|  | /* | 
|  | *  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 <http://www.gnu.org/licenses/>. | 
|  | */ | 
|  |  | 
|  | #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; | 
|  | } | 
|  |  | 
|  | /* undocumented 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; | 
|  | } | 
|  |  | 
|  | /* undocumented 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; | 
|  | } | 
|  |  | 
|  | /* undocumented 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 */ |