/* * stat related system call shims and definitions * * Copyright (c) 2013 Stacey D. Son * * 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, see . */ #ifndef BSD_USER_FREEBSD_OS_STAT_H #define BSD_USER_FREEBSD_OS_STAT_H int freebsd11_stat(const char *path, struct freebsd11_stat *stat); __sym_compat(stat, freebsd11_stat, FBSD_1.0); int freebsd11_lstat(const char *path, struct freebsd11_stat *stat); __sym_compat(lstat, freebsd11_lstat, FBSD_1.0); int freebsd11_fstat(int fd, struct freebsd11_stat *stat); __sym_compat(fstat, freebsd11_fstat, FBSD_1.0); int freebsd11_fstatat(int fd, const char *path, struct freebsd11_stat *stat, int flag); __sym_compat(fstatat, freebsd11_fstatat, FBSD_1.1); int freebsd11_fhstat(const fhandle_t *fhandle, struct freebsd11_stat *stat); __sym_compat(fhstat, freebsd11_fhstat, FBSD_1.0); int freebsd11_getfsstat(struct freebsd11_statfs *buf, long bufsize, int mode); __sym_compat(getfsstat, freebsd11_getfsstat, FBSD_1.0); int freebsd11_fhstatfs(const fhandle_t *fhandle, struct freebsd11_statfs * buf); __sym_compat(fhstatfs, freebsd11_fhstatfs, FBSD_1.0); int freebsd11_statfs(const char *path, struct freebsd11_statfs *buf); __sym_compat(statfs, freebsd11_statfs, FBSD_1.0); int freebsd11_fstatfs(int fd, struct freebsd11_statfs *buf); __sym_compat(fstatfs, freebsd11_fstatfs, FBSD_1.0); ssize_t freebsd11_getdirentries(int fd, char *buf, size_t nbytes, off_t *basep); __sym_compat(getdirentries, freebsd11_getdirentries, FBSD_1.0); ssize_t freebsd11_getdents(int fd, char *buf, size_t nbytes); __sym_compat(getdents, freebsd11_getdents, FBSD_1.0); /* undocumented nstat system calls */ int freebsd11_nstat(const char *path, struct freebsd11_stat *sb); __sym_compat(nstat, freebsd11_nstat, FBSD_1.0); int freebsd11_nlstat(const char *path, struct freebsd11_stat *sb); __sym_compat(nlstat, freebsd11_nlstat, FBSD_1.0); int freebsd11_nfstat(int fd, struct freebsd11_stat *sb); __sym_compat(nfstat, freebsd11_nfstat, FBSD_1.0); /* stat(2) */ static inline abi_long do_freebsd11_stat(abi_long arg1, abi_long arg2) { abi_long ret; void *p; struct freebsd11_stat st; LOCK_PATH(p, arg1); ret = get_errno(freebsd11_stat(path(p), &st)); UNLOCK_PATH(p, arg1); if (!is_error(ret)) { ret = h2t_freebsd11_stat(arg2, &st); } return ret; } /* lstat(2) */ static inline abi_long do_freebsd11_lstat(abi_long arg1, abi_long arg2) { abi_long ret; void *p; struct freebsd11_stat st; LOCK_PATH(p, arg1); ret = get_errno(freebsd11_lstat(path(p), &st)); UNLOCK_PATH(p, arg1); if (!is_error(ret)) { ret = h2t_freebsd11_stat(arg2, &st); } return ret; } /* fstat(2) */ static inline abi_long do_freebsd11_fstat(abi_long arg1, abi_long arg2) { abi_long ret; struct freebsd11_stat st; ret = get_errno(freebsd11_fstat(arg1, &st)); if (!is_error(ret)) { ret = h2t_freebsd11_stat(arg2, &st); } return ret; } /* fstat(2) */ static inline abi_long do_freebsd_fstat(abi_long arg1, abi_long arg2) { abi_long ret; struct stat st; ret = get_errno(fstat(arg1, &st)); if (!is_error(ret)) { ret = h2t_freebsd_stat(arg2, &st); } return ret; } /* fstatat(2) */ static inline abi_long do_freebsd11_fstatat(abi_long arg1, abi_long arg2, abi_long arg3, abi_long arg4) { abi_long ret; void *p; struct freebsd11_stat st; LOCK_PATH(p, arg2); ret = get_errno(freebsd11_fstatat(arg1, p, &st, arg4)); UNLOCK_PATH(p, arg2); if (!is_error(ret) && arg3) { ret = h2t_freebsd11_stat(arg3, &st); } return ret; } /* fstatat(2) */ static inline abi_long do_freebsd_fstatat(abi_long arg1, abi_long arg2, abi_long arg3, abi_long arg4) { abi_long ret; void *p; struct stat st; LOCK_PATH(p, arg2); ret = get_errno(fstatat(arg1, p, &st, arg4)); UNLOCK_PATH(p, arg2); if (!is_error(ret) && arg3) { ret = h2t_freebsd_stat(arg3, &st); } return ret; } /* undocummented nstat(char *path, struct nstat *ub) syscall */ static abi_long do_freebsd11_nstat(abi_long arg1, abi_long arg2) { abi_long ret; void *p; struct freebsd11_stat st; LOCK_PATH(p, arg1); ret = get_errno(freebsd11_nstat(path(p), &st)); UNLOCK_PATH(p, arg1); if (!is_error(ret)) { ret = h2t_freebsd11_nstat(arg2, &st); } return ret; } /* undocummented nfstat(int fd, struct nstat *sb) syscall */ static abi_long do_freebsd11_nfstat(abi_long arg1, abi_long arg2) { abi_long ret; struct freebsd11_stat st; ret = get_errno(freebsd11_nfstat(arg1, &st)); if (!is_error(ret)) { ret = h2t_freebsd11_nstat(arg2, &st); } return ret; } /* undocummented nlstat(char *path, struct nstat *ub) syscall */ static abi_long do_freebsd11_nlstat(abi_long arg1, abi_long arg2) { abi_long ret; void *p; struct freebsd11_stat st; LOCK_PATH(p, arg1); ret = get_errno(freebsd11_nlstat(path(p), &st)); UNLOCK_PATH(p, arg1); if (!is_error(ret)) { ret = h2t_freebsd11_nstat(arg2, &st); } return ret; } /* getfh(2) */ static abi_long do_freebsd_getfh(abi_long arg1, abi_long arg2) { abi_long ret; void *p; fhandle_t host_fh; LOCK_PATH(p, arg1); ret = get_errno(getfh(path(p), &host_fh)); UNLOCK_PATH(p, arg1); if (is_error(ret)) { return ret; } return h2t_freebsd_fhandle(arg2, &host_fh); } /* lgetfh(2) */ static inline abi_long do_freebsd_lgetfh(abi_long arg1, abi_long arg2) { abi_long ret; void *p; fhandle_t host_fh; LOCK_PATH(p, arg1); ret = get_errno(lgetfh(path(p), &host_fh)); UNLOCK_PATH(p, arg1); if (is_error(ret)) { return ret; } return h2t_freebsd_fhandle(arg2, &host_fh); } /* fhopen(2) */ static inline abi_long do_freebsd_fhopen(abi_long arg1, abi_long arg2) { abi_long ret; fhandle_t host_fh; ret = t2h_freebsd_fhandle(&host_fh, arg1); if (is_error(ret)) { return ret; } return get_errno(fhopen(&host_fh, arg2)); } /* fhstat(2) */ static inline abi_long do_freebsd11_fhstat(abi_long arg1, abi_long arg2) { abi_long ret; fhandle_t host_fh; struct freebsd11_stat host_sb; ret = t2h_freebsd_fhandle(&host_fh, arg1); if (is_error(ret)) { return ret; } ret = get_errno(freebsd11_fhstat(&host_fh, &host_sb)); if (is_error(ret)) { return ret; } return h2t_freebsd11_stat(arg2, &host_sb); } /* fhstat(2) */ static inline abi_long do_freebsd_fhstat(abi_long arg1, abi_long arg2) { abi_long ret; fhandle_t host_fh; struct stat host_sb; ret = t2h_freebsd_fhandle(&host_fh, arg1); if (is_error(ret)) { return ret; } ret = get_errno(fhstat(&host_fh, &host_sb)); if (is_error(ret)) { return ret; } return h2t_freebsd_stat(arg2, &host_sb); } /* fhstatfs(2) */ static inline abi_long do_freebsd11_fhstatfs(abi_ulong target_fhp_addr, abi_ulong target_stfs_addr) { abi_long ret; fhandle_t host_fh; struct freebsd11_statfs host_stfs; ret = t2h_freebsd_fhandle(&host_fh, target_fhp_addr); if (is_error(ret)) { return ret; } ret = get_errno(freebsd11_fhstatfs(&host_fh, &host_stfs)); if (is_error(ret)) { return ret; } return h2t_freebsd11_statfs(target_stfs_addr, &host_stfs); } /* fhstatfs(2) */ static inline abi_long do_freebsd_fhstatfs(abi_ulong target_fhp_addr, abi_ulong target_stfs_addr) { abi_long ret; fhandle_t host_fh; struct statfs host_stfs; ret = t2h_freebsd_fhandle(&host_fh, target_fhp_addr); if (is_error(ret)) { return ret; } ret = get_errno(fhstatfs(&host_fh, &host_stfs)); if (is_error(ret)) { return ret; } return h2t_freebsd_statfs(target_stfs_addr, &host_stfs); } /* statfs(2) */ static inline abi_long do_freebsd11_statfs(abi_long arg1, abi_long arg2) { abi_long ret; void *p; struct freebsd11_statfs host_stfs; LOCK_PATH(p, arg1); ret = get_errno(freebsd11_statfs(path(p), &host_stfs)); UNLOCK_PATH(p, arg1); if (is_error(ret)) { return ret; } return h2t_freebsd11_statfs(arg2, &host_stfs); } /* statfs(2) */ static inline abi_long do_freebsd_statfs(abi_long arg1, abi_long arg2) { abi_long ret; void *p; struct statfs host_stfs; LOCK_PATH(p, arg1); ret = get_errno(statfs(path(p), &host_stfs)); UNLOCK_PATH(p, arg1); if (is_error(ret)) { return ret; } return h2t_freebsd_statfs(arg2, &host_stfs); } /* fstatfs(2) */ static inline abi_long do_freebsd11_fstatfs(abi_long fd, abi_ulong target_addr) { abi_long ret; struct freebsd11_statfs host_stfs; ret = get_errno(freebsd11_fstatfs(fd, &host_stfs)); if (is_error(ret)) { return ret; } return h2t_freebsd11_statfs(target_addr, &host_stfs); } /* fstatfs(2) */ static inline abi_long do_freebsd_fstatfs(abi_long fd, abi_ulong target_addr) { abi_long ret; struct statfs host_stfs; ret = get_errno(fstatfs(fd, &host_stfs)); if (is_error(ret)) { return ret; } return h2t_freebsd_statfs(target_addr, &host_stfs); } /* getfsstat(2) */ static inline abi_long do_freebsd11_getfsstat(abi_ulong target_addr, abi_long bufsize, abi_long flags) { abi_long ret; struct freebsd11_statfs *host_stfs; int count; long host_bufsize; count = bufsize / sizeof(struct target_freebsd11_statfs); /* if user buffer is NULL then return number of mounted FS's */ if (target_addr == 0 || count == 0) { return get_errno(freebsd11_getfsstat(NULL, 0, flags)); } /* XXX check count to be reasonable */ host_bufsize = sizeof(struct freebsd11_statfs) * count; host_stfs = alloca(host_bufsize); if (!host_stfs) { return -TARGET_EINVAL; } ret = count = get_errno(freebsd11_getfsstat(host_stfs, host_bufsize, flags)); if (is_error(ret)) { return ret; } while (count--) { if (h2t_freebsd11_statfs((target_addr + (count * sizeof(struct target_freebsd11_statfs))), &host_stfs[count])) { return -TARGET_EFAULT; } } return ret; } /* getfsstat(2) */ static inline abi_long do_freebsd_getfsstat(abi_ulong target_addr, abi_long bufsize, abi_long flags) { abi_long ret; struct statfs *host_stfs; int count; long host_bufsize; count = bufsize / sizeof(struct target_statfs); /* if user buffer is NULL then return number of mounted FS's */ if (target_addr == 0 || count == 0) { return get_errno(freebsd11_getfsstat(NULL, 0, flags)); } /* XXX check count to be reasonable */ host_bufsize = sizeof(struct statfs) * count; host_stfs = alloca(host_bufsize); if (!host_stfs) { return -TARGET_EINVAL; } ret = count = get_errno(getfsstat(host_stfs, host_bufsize, flags)); if (is_error(ret)) { return ret; } while (count--) { if (h2t_freebsd_statfs((target_addr + (count * sizeof(struct target_statfs))), &host_stfs[count])) { return -TARGET_EFAULT; } } return ret; } /* getdents(2) */ static inline abi_long do_freebsd11_getdents(abi_long arg1, abi_ulong arg2, abi_long nbytes) { abi_long ret; struct freebsd11_dirent *dirp; dirp = lock_user(VERIFY_WRITE, arg2, nbytes, 0); if (dirp == NULL) { return -TARGET_EFAULT; } ret = get_errno(freebsd11_getdents(arg1, (char *)dirp, nbytes)); if (!is_error(ret)) { struct freebsd11_dirent *de; int len = ret; int reclen; de = dirp; while (len > 0) { reclen = de->d_reclen; if (reclen > len) { return -TARGET_EFAULT; } de->d_reclen = tswap16(reclen); de->d_fileno = tswap32(de->d_fileno); len -= reclen; } } return ret; } /* getdirecentries(2) */ static inline abi_long do_freebsd11_getdirentries(abi_long arg1, abi_ulong arg2, abi_long nbytes, abi_ulong arg4) { abi_long ret; struct freebsd11_dirent *dirp; long basep; dirp = lock_user(VERIFY_WRITE, arg2, nbytes, 0); if (dirp == NULL) { return -TARGET_EFAULT; } ret = get_errno(freebsd11_getdirentries(arg1, (char *)dirp, nbytes, &basep)); if (!is_error(ret)) { struct freebsd11_dirent *de; int len = ret; int reclen; de = dirp; while (len > 0) { reclen = de->d_reclen; if (reclen > len) { return -TARGET_EFAULT; } de->d_reclen = tswap16(reclen); de->d_fileno = tswap32(de->d_fileno); len -= reclen; de = (struct freebsd11_dirent *)((void *)de + reclen); } } unlock_user(dirp, arg2, ret); if (arg4) { if (put_user(basep, arg4, abi_ulong)) { return -TARGET_EFAULT; } } return ret; } /* getdirecentries(2) */ static inline abi_long do_freebsd_getdirentries(abi_long arg1, abi_ulong arg2, abi_long nbytes, abi_ulong arg4) { abi_long ret; struct dirent *dirp; long basep; dirp = lock_user(VERIFY_WRITE, arg2, nbytes, 0); if (dirp == NULL) { return -TARGET_EFAULT; } ret = get_errno(getdirentries(arg1, (char *)dirp, nbytes, &basep)); if (!is_error(ret)) { struct dirent *de; int len = ret; int reclen; de = dirp; while (len > 0) { reclen = de->d_reclen; if (reclen > len) { return -TARGET_EFAULT; } de->d_fileno = tswap64(de->d_fileno); de->d_off = tswap64(de->d_off); de->d_reclen = tswap16(de->d_reclen); de->d_namlen = tswap16(de->d_namlen); len -= reclen; de = (struct dirent *)((void *)de + reclen); } } unlock_user(dirp, arg2, ret); if (arg4) { if (put_user(basep, arg4, abi_ulong)) { return -TARGET_EFAULT; } } return ret; } /* fcntl(2) */ static inline abi_long do_freebsd_fcntl(abi_long arg1, abi_long arg2, abi_ulong arg3) { abi_long ret; int host_cmd; struct flock fl; struct target_freebsd_flock *target_fl; host_cmd = target_to_host_fcntl_cmd(arg2); if (host_cmd < 0) { return host_cmd; } switch (arg2) { case TARGET_F_GETLK: if (!lock_user_struct(VERIFY_READ, target_fl, arg3, 1)) { return -TARGET_EFAULT; } __get_user(fl.l_type, &target_fl->l_type); __get_user(fl.l_whence, &target_fl->l_whence); __get_user(fl.l_start, &target_fl->l_start); __get_user(fl.l_len, &target_fl->l_len); __get_user(fl.l_pid, &target_fl->l_pid); __get_user(fl.l_sysid, &target_fl->l_sysid); unlock_user_struct(target_fl, arg3, 0); ret = get_errno(safe_fcntl(arg1, host_cmd, &fl)); if (!is_error(ret)) { if (!lock_user_struct(VERIFY_WRITE, target_fl, arg3, 0)) { return -TARGET_EFAULT; } __put_user(fl.l_type, &target_fl->l_type); __put_user(fl.l_whence, &target_fl->l_whence); __put_user(fl.l_start, &target_fl->l_start); __put_user(fl.l_len, &target_fl->l_len); __put_user(fl.l_pid, &target_fl->l_pid); __put_user(fl.l_sysid, &target_fl->l_sysid); unlock_user_struct(target_fl, arg3, 1); } break; case TARGET_F_SETLK: case TARGET_F_SETLKW: if (!lock_user_struct(VERIFY_READ, target_fl, arg3, 1)) { return -TARGET_EFAULT; } __get_user(fl.l_type, &target_fl->l_type); __get_user(fl.l_whence, &target_fl->l_whence); __get_user(fl.l_start, &target_fl->l_start); __get_user(fl.l_len, &target_fl->l_len); __get_user(fl.l_pid, &target_fl->l_pid); __get_user(fl.l_sysid, &target_fl->l_sysid); unlock_user_struct(target_fl, arg3, 0); ret = get_errno(safe_fcntl(arg1, host_cmd, &fl)); break; case TARGET_F_DUPFD: case TARGET_F_DUP2FD: case TARGET_F_GETOWN: case TARGET_F_SETOWN: case TARGET_F_GETFD: case TARGET_F_SETFD: case TARGET_F_GETFL: case TARGET_F_SETFL: case TARGET_F_READAHEAD: case TARGET_F_RDAHEAD: case TARGET_F_ADD_SEALS: case TARGET_F_GET_SEALS: default: ret = get_errno(safe_fcntl(arg1, host_cmd, arg3)); break; } return ret; } #if defined(__FreeBSD_version) && __FreeBSD_version >= 1300080 extern int __realpathat(int fd, const char *path, char *buf, size_t size, int flags); /* https://svnweb.freebsd.org/base?view=revision&revision=358172 */ /* no man page */ static inline abi_long do_freebsd_realpathat(abi_long arg1, abi_long arg2, abi_long arg3, abi_long arg4, abi_long arg5) { abi_long ret; void *p, *b; LOCK_PATH(p, arg2); b = lock_user(VERIFY_WRITE, arg3, arg4, 0); if (b == NULL) { UNLOCK_PATH(p, arg2); return -TARGET_EFAULT; } ret = get_errno(__realpathat(arg1, p, b, arg4, arg5)); UNLOCK_PATH(p, arg2); unlock_user(b, arg3, ret); return ret; } #endif #endif /* BSD_USER_FREEBSD_OS_STAT_H */