/* * CDDL HEADER START * * The contents of this file are subject to the terms of the * Common Development and Distribution License (the "License"). * You may not use this file except in compliance with the License. * * You can obtain a copy of the license at usr/src/OPENSOLARIS.LICENSE * or http://www.opensolaris.org/os/licensing. * See the License for the specific language governing permissions * and limitations under the License. * * When distributing Covered Code, include this CDDL HEADER in each * file and include the License file at usr/src/OPENSOLARIS.LICENSE. * If applicable, add the following below this CDDL HEADER, with the * fields enclosed by brackets "[]" replaced with your own identifying * information: Portions Copyright [yyyy] [name of copyright owner] * * CDDL HEADER END */ /* * Copyright 2006 Sun Microsystems, Inc. All rights reserved. * Use is subject to license terms. */ /* Copyright (c) 1983, 1984, 1985, 1986, 1987, 1988, 1989 AT&T */ /* All Rights Reserved */ /* * Portions of this source code were derived from Berkeley 4.3 BSD * under license from the Regents of the University of California. */ #pragma ident "%Z%%M% %I% %E% SMI" /* * Get file system statistics (statvfs and fstatvfs). */ #include <sys/types.h> #include <sys/inttypes.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/statvfs.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> #define STATVFSCOPY(dst, src) \ (dst)->f_bsize = (src)->f_bsize; \ (dst)->f_frsize = (src)->f_frsize; \ (dst)->f_blocks = (src)->f_blocks; \ (dst)->f_bfree = (src)->f_bfree; \ (dst)->f_bavail = (src)->f_bavail; \ (dst)->f_files = (src)->f_files; \ (dst)->f_ffree = (src)->f_ffree; \ (dst)->f_favail = (src)->f_favail; \ (dst)->f_fsid = (src)->f_fsid; \ bcopy((src)->f_basetype, (dst)->f_basetype, \ sizeof ((dst)->f_basetype)); \ (dst)->f_flag = (src)->f_flag; \ (dst)->f_namemax = (src)->f_namemax; \ bcopy((src)->f_fstr, (dst)->f_fstr, \ sizeof ((dst)->f_fstr)) /* * Common routines for statvfs and fstatvfs. */ static int cstatvfs32(struct vfs *vfsp, struct statvfs32 *ubp) { struct statvfs64 ds64; struct statvfs32 ds32; int error; #if !defined(lint) ASSERT32(sizeof (struct statvfs) == sizeof (struct statvfs32)); ASSERT32(sizeof (struct statvfs64) == sizeof (struct statvfs64_32)); #endif bzero(&ds64, sizeof (ds64)); if ((error = VFS_STATVFS(vfsp, &ds64)) != 0) return (error); /* * VFS_STATVFS can return data that is incompatible with the space * available the 32-bit statvfs structure. Check here to see if * it will fit into the 32-bit structure, if not, return EOVERFLOW. * * The check for -1 is because some file systems return -1 in the * fields that are irrelevant or nonessential, and we do not want * to return EOVERFLOW for them. For example: df is expected to * show -1 in the output for some of these fields on NFS mounted * filesystems. */ if (ds64.f_files == (fsfilcnt64_t)-1) ds64.f_files = UINT32_MAX; if (ds64.f_ffree == (fsfilcnt64_t)-1) ds64.f_ffree = UINT32_MAX; if (ds64.f_favail == (fsfilcnt64_t)-1) ds64.f_favail = UINT32_MAX; if (ds64.f_bavail == (fsblkcnt64_t)-1) ds64.f_bavail = UINT32_MAX; if (ds64.f_bfree == (fsblkcnt64_t)-1) ds64.f_bfree = UINT32_MAX; if (ds64.f_blocks > UINT32_MAX || ds64.f_bfree > UINT32_MAX || ds64.f_bavail > UINT32_MAX || ds64.f_files > UINT32_MAX || ds64.f_ffree > UINT32_MAX || ds64.f_favail > UINT32_MAX) return (EOVERFLOW); #ifdef _LP64 /* * On the 64-bit kernel, even these fields grow to 64-bit * quantities in the statvfs64 structure. */ if (ds64.f_namemax == (ulong_t)-1l) ds64.f_namemax = UINT32_MAX; if (ds64.f_bsize > UINT32_MAX || ds64.f_frsize > UINT32_MAX || ds64.f_fsid > UINT32_MAX || ds64.f_flag > UINT32_MAX || ds64.f_namemax > UINT32_MAX) return (EOVERFLOW); #endif bzero(&ds32, sizeof (ds32)); STATVFSCOPY(&ds32, &ds64); if (copyout(&ds32, ubp, sizeof (ds32)) != 0) return (EFAULT); return (0); } static int cstatvfs64(struct vfs *vfsp, struct statvfs64 *ubp) { struct statvfs64 ds64; int error; #if !defined(lint) ASSERT64(sizeof (struct statvfs) == sizeof (struct statvfs64)); #endif bzero(&ds64, sizeof (ds64)); if ((error = VFS_STATVFS(vfsp, &ds64)) != 0) return (error); if (copyout(&ds64, ubp, sizeof (ds64)) != 0) return (EFAULT); return (0); } /* * Native system calls */ int statvfs(char *fname, struct statvfs *sbp) { 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)); } #ifdef _LP64 error = cstatvfs64(vp->v_vfsp, (struct statvfs64 *)sbp); #else error = cstatvfs32(vp->v_vfsp, (struct statvfs32 *)sbp); #endif VN_RELE(vp); if (error) { if ((error == ESTALE) && fs_need_estale_retry(estale_retry++)) goto lookup; return (set_errno(error)); } return (0); } int fstatvfs(int fdes, struct statvfs *sbp) { struct file *fp; int error; if ((fp = getf(fdes)) == NULL) return (set_errno(EBADF)); #ifdef _LP64 error = cstatvfs64(fp->f_vnode->v_vfsp, (struct statvfs64 *)sbp); #else error = cstatvfs32(fp->f_vnode->v_vfsp, (struct statvfs32 *)sbp); #endif releasef(fdes); if (error) return (set_errno(error)); return (0); } #if defined(_ILP32) /* * Large File system calls. * * (We deliberately don't have special "large file" system calls in the * 64-bit kernel -- we just use the native versions, since they're just * as functional.) */ int statvfs64(char *fname, struct statvfs64 *sbp) { 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)); } error = cstatvfs64(vp->v_vfsp, sbp); VN_RELE(vp); if (error) { if ((error == ESTALE) && fs_need_estale_retry(estale_retry++)) goto lookup; return (set_errno(error)); } return (0); } int fstatvfs64(int fdes, struct statvfs64 *sbp) { struct file *fp; int error; if ((fp = getf(fdes)) == NULL) return (set_errno(EBADF)); error = cstatvfs64(fp->f_vnode->v_vfsp, sbp); releasef(fdes); if (error) return (set_errno(error)); return (0); } #endif /* _ILP32 */ #ifdef _SYSCALL32_IMPL static int cstatvfs64_32(struct vfs *vfsp, struct statvfs64_32 *ubp) { struct statvfs64 ds64; struct statvfs64_32 ds64_32; int error; bzero(&ds64, sizeof (ds64)); if ((error = VFS_STATVFS(vfsp, &ds64)) != 0) return (error); /* * On the 64-bit kernel, even these fields grow to 64-bit * quantities in the statvfs64 structure. */ if (ds64.f_namemax == (ulong_t)-1l) ds64.f_namemax = UINT32_MAX; if (ds64.f_bsize > UINT32_MAX || ds64.f_frsize > UINT32_MAX || ds64.f_fsid > UINT32_MAX || ds64.f_flag > UINT32_MAX || ds64.f_namemax > UINT32_MAX) return (EOVERFLOW); STATVFSCOPY(&ds64_32, &ds64); if (copyout(&ds64_32, ubp, sizeof (ds64_32)) != 0) return (EFAULT); return (0); } /* * ILP32 "small file" system calls on LP64 kernel */ int statvfs32(char *fname, struct statvfs32 *sbp) { 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)); } error = cstatvfs32(vp->v_vfsp, sbp); VN_RELE(vp); if (error) { if ((error == ESTALE) && fs_need_estale_retry(estale_retry++)) goto lookup; return (set_errno(error)); } return (0); } int fstatvfs32(int fdes, struct statvfs32 *sbp) { struct file *fp; int error; if ((fp = getf(fdes)) == NULL) return (set_errno(EBADF)); error = cstatvfs32(fp->f_vnode->v_vfsp, sbp); releasef(fdes); if (error) return (set_errno(error)); return (0); } /* * ILP32 Large File system calls on LP64 kernel */ int statvfs64_32(char *fname, struct statvfs64_32 *sbp) { 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)); } error = cstatvfs64_32(vp->v_vfsp, sbp); VN_RELE(vp); if (error) { if ((error == ESTALE) && fs_need_estale_retry(estale_retry++)) goto lookup; return (set_errno(error)); } return (0); } int fstatvfs64_32(int fdes, struct statvfs64_32 *sbp) { struct file *fp; int error; if ((fp = getf(fdes)) == NULL) return (set_errno(EBADF)); error = cstatvfs64_32(fp->f_vnode->v_vfsp, sbp); releasef(fdes); if (error) return (set_errno(error)); return (0); } #endif /* _SYSCALL32_IMPL */