#include <sys/inttypes.h>
#include <sys/types.h>
#include <sys/t_lock.h>
#include <sys/param.h>
#include <sys/errno.h>
#include <sys/fstyp.h>
#include <sys/systm.h>
#include <sys/vfs.h>
#include <sys/statfs.h>
#include <sys/vnode.h>
#include <sys/file.h>
#include <sys/cmn_err.h>
#include <sys/debug.h>
#include <sys/pathname.h>
#include <vm/page.h>
#include <fs/fs_subr.h>
#if defined(_SYSCALL32_IMPL) || defined(_ILP32)
static int cstatfs(struct vfs *, struct statfs32 *, int);
int
statfs32(char *fname, struct statfs32 *sbp, int32_t len, int32_t fstyp)
{
vnode_t *vp;
int error;
int estale_retry = 0;
lookup:
if (error = lookupname(fname, UIO_USERSPACE, FOLLOW, NULLVPP, &vp)) {
if ((error == ESTALE) && fs_need_estale_retry(estale_retry++))
goto lookup;
return (set_errno(error));
}
if (fstyp != 0)
error = EINVAL;
else
error = cstatfs(vp->v_vfsp, sbp, len);
VN_RELE(vp);
if (error) {
if ((error == ESTALE) && fs_need_estale_retry(estale_retry++))
goto lookup;
return (set_errno(error));
}
return (0);
}
int
fstatfs32(int32_t fdes, struct statfs32 *sbp, int32_t len, int32_t fstyp)
{
struct file *fp;
int error;
if (fstyp != 0)
return (set_errno(EINVAL));
if ((fp = getf(fdes)) == NULL)
return (set_errno(EBADF));
error = cstatfs(fp->f_vnode->v_vfsp, sbp, len);
releasef(fdes);
if (error)
return (set_errno(error));
return (0);
}
static int
cstatfs(struct vfs *vfsp, struct statfs32 *sbp, int len)
{
struct statfs32 sfs;
struct statvfs64 svfs;
int error, i;
char *cp, *cp2;
struct vfssw *vswp;
if (len < 0 || len > sizeof (struct statfs))
return (EINVAL);
if (error = VFS_STATVFS(vfsp, &svfs))
return (error);
if (svfs.f_blocks > UINT32_MAX || svfs.f_bfree > UINT32_MAX ||
svfs.f_files > UINT32_MAX || svfs.f_ffree > UINT32_MAX)
return (EOVERFLOW);
bzero(&sfs, sizeof (sfs));
sfs.f_bsize = svfs.f_bsize;
sfs.f_frsize = (svfs.f_frsize == svfs.f_bsize) ? 0 : svfs.f_frsize;
sfs.f_blocks = svfs.f_blocks * (svfs.f_frsize / 512);
sfs.f_bfree = svfs.f_bfree * (svfs.f_frsize / 512);
sfs.f_files = svfs.f_files;
sfs.f_ffree = svfs.f_ffree;
cp = svfs.f_fstr;
cp2 = sfs.f_fname;
i = 0;
while (i++ < sizeof (sfs.f_fname))
if (*cp != '\0')
*cp2++ = *cp++;
else
*cp2++ = '\0';
while (*cp != '\0' &&
i++ < (sizeof (svfs.f_fstr) - sizeof (sfs.f_fpack)))
cp++;
(void) strncpy(sfs.f_fpack, cp + 1, sizeof (sfs.f_fpack));
if ((vswp = vfs_getvfssw(svfs.f_basetype)) == NULL)
sfs.f_fstyp = 0;
else {
sfs.f_fstyp = vswp - vfssw;
vfs_unrefvfssw(vswp);
}
if (copyout(&sfs, sbp, len))
return (EFAULT);
return (0);
}
#endif