Skip to content

Commit

Permalink
Support statvfs on API levels before 19.
Browse files Browse the repository at this point in the history
Split statfs and statvfs. The former has been available forever, and the
latter is implemented in terms of the former. The implementation has
been moved into headers so that it can be used at low API levels.

There's no reason for any Android or Linux code to use statvfs rather
than statfs, but code that needs to build on Darwin too will want to use
statvfs because Darwin's statfs is very spartan.

Bug: android/ndk#609
Test: treehugger
Change-Id: Icf3d5723a260099fddb2d9f902e3047b0f041647
  • Loading branch information
enh-google committed Aug 30, 2019
1 parent 9e544d3 commit 06c99a3
Show file tree
Hide file tree
Showing 6 changed files with 289 additions and 70 deletions.
3 changes: 2 additions & 1 deletion libc/Android.bp
Original file line number Diff line number Diff line change
Expand Up @@ -1133,7 +1133,6 @@ cc_library_static {
"bionic/sigprocmask.cpp",
"bionic/spawn.cpp",
"bionic/stat.cpp",
"bionic/statvfs.cpp",
"bionic/stdlib_l.cpp",
"bionic/strchrnul.cpp",
"bionic/strerror.cpp",
Expand All @@ -1150,6 +1149,8 @@ cc_library_static {
"bionic/sys_sem.cpp",
"bionic/sys_shm.cpp",
"bionic/sys_signalfd.cpp",
"bionic/sys_statfs.cpp",
"bionic/sys_statvfs.cpp",
"bionic/sys_time.cpp",
"bionic/sysinfo.cpp",
"bionic/syslog.cpp",
Expand Down
48 changes: 6 additions & 42 deletions libc/bionic/statvfs.cpp → libc/bionic/sys_statfs.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -14,12 +14,11 @@
* limitations under the License.
*/

#include <sys/statvfs.h>

#include <sys/statfs.h>

// Paper over the fact that 32-bit kernels use fstatfs64/statfs64 with an extra argument,
// but 64-bit kernels don't have the "64" bit suffix or the extra size_t argument.
// Paper over the fact that 32-bit kernels use fstatfs64/statfs64 with
// an extra argument, but 64-bit kernels don't have the "64" bit suffix or
// the extra size_t argument.
#if defined(__LP64__)
extern "C" int __fstatfs(int, struct statfs*);
extern "C" int __statfs(const char*, struct statfs*);
Expand All @@ -30,24 +29,11 @@ extern "C" int __fstatfs64(int, size_t, struct statfs*);
extern "C" int __statfs64(const char*, size_t, struct statfs*);
#endif

// The kernel sets a private ST_VALID flag to signal to the C library whether the
// f_flags field is valid. This flag should not be exposed to users of the C library.
// The kernel sets a private ST_VALID flag to signal to the C library
// whether the f_flags field is valid. This flag should not be exposed to
// users of the C library.
#define ST_VALID 0x0020

static void __statfs_to_statvfs(const struct statfs& in, struct statvfs* out) {
out->f_bsize = in.f_bsize;
out->f_frsize = in.f_frsize;
out->f_blocks = in.f_blocks;
out->f_bfree = in.f_bfree;
out->f_bavail = in.f_bavail;
out->f_files = in.f_files;
out->f_ffree = in.f_ffree;
out->f_favail = in.f_ffree;
out->f_fsid = in.f_fsid.__val[0] | (static_cast<uint64_t>(in.f_fsid.__val[1]) << 32);
out->f_flag = in.f_flags;
out->f_namemax = in.f_namelen;
}

int fstatfs(int fd, struct statfs* result) {
int rc = __fstatfs64(fd, sizeof(*result), result);
if (rc != 0) {
Expand All @@ -67,25 +53,3 @@ int statfs(const char* path, struct statfs* result) {
return 0;
}
__strong_alias(statfs64, statfs);

int statvfs(const char* path, struct statvfs* result) {
struct statfs tmp;
int rc = statfs(path, &tmp);
if (rc != 0) {
return rc;
}
__statfs_to_statvfs(tmp, result);
return 0;
}
__strong_alias(statvfs64, statvfs);

int fstatvfs(int fd, struct statvfs* result) {
struct statfs tmp;
int rc = fstatfs(fd, &tmp);
if (rc != 0) {
return rc;
}
__statfs_to_statvfs(tmp, result);
return 0;
}
__strong_alias(fstatvfs64, fstatvfs);
36 changes: 36 additions & 0 deletions libc/bionic/sys_statvfs.cpp
Original file line number Diff line number Diff line change
@@ -0,0 +1,36 @@
/*
* Copyright (C) 2013 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/

#include <sys/statvfs.h>

// libc++ uses statvfs (for Darwin compatibility), but on Linux statvfs is
// just another name for statfs, so it didn't arrive until API level 19. We
// make the implementation available as inlines to support std::filesystem
// for NDK users (see https://github.com/android-ndk/ndk/issues/609).

// Ensure we get all the inlines.
#undef __ANDROID_API__
#define __ANDROID_API__ 16

#define __BIONIC_SYS_STATVFS_INLINE /* Out of line. */
#include <bits/sys_statvfs_inlines.h>

// Historically we provided actual symbols for statvfs64 and fstatvfs64.
// They're not particularly useful, but we can't take them away.

__strong_alias(statvfs64, statvfs);

__strong_alias(fstatvfs64, fstatvfs);
43 changes: 43 additions & 0 deletions libc/include/android/legacy_sys_statvfs_inlines.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,43 @@
/*
* Copyright (C) 2019 The Android Open Source Project
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*/

#pragma once

/**
* @file legacy_sys_statvfs_inlines.h
* @brief Inline definitions of statvfs/fstatvfs for old API levels.
*/

#include <sys/cdefs.h>

#if __ANDROID_API__ < 21

#define __BIONIC_SYS_STATVFS_INLINE static __inline
#include <bits/sys_statvfs_inlines.h>

#endif
95 changes: 95 additions & 0 deletions libc/include/bits/sys_statvfs_inlines.h
Original file line number Diff line number Diff line change
@@ -0,0 +1,95 @@
/*
* Copyright (C) 2019 The Android Open Source Project
* All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in
* the documentation and/or other materials provided with the
* distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
* COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS
* OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED
* AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT
* OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*/

#pragma once

#include <sys/cdefs.h>
#include <sys/statfs.h>
#include <sys/statvfs.h>

#if defined(__BIONIC_SYS_STATVFS_INLINE)

__BEGIN_DECLS

#if __ANDROID_API__ < 19

static __inline void __bionic_statfs_to_statvfs(const struct statfs* __in,
struct statvfs* __out) {
__out->f_bsize = __in->f_bsize;
__out->f_frsize = __in->f_frsize;
__out->f_blocks = __in->f_blocks;
__out->f_bfree = __in->f_bfree;
__out->f_bavail = __in->f_bavail;
__out->f_files = __in->f_files;
__out->f_ffree = __in->f_ffree;
__out->f_favail = __in->f_ffree;
__out->f_fsid = __in->f_fsid.__val[0] |
__BIONIC_CAST(static_cast, uint64_t, __in->f_fsid.__val[1]) << 32;
__out->f_flag = __in->f_flags;
__out->f_namemax = __in->f_namelen;
}

__BIONIC_SYS_STATVFS_INLINE int statvfs(const char* __path,
struct statvfs* __result) {
struct statfs __tmp;
int __rc = statfs(__path, &__tmp);
if (__rc != 0) return __rc;
__bionic_statfs_to_statvfs(&__tmp, __result);
return 0;
}

__BIONIC_SYS_STATVFS_INLINE int fstatvfs(int __fd,
struct statvfs* __result) {
struct statfs __tmp;
int __rc = fstatfs(__fd, &__tmp);
if (__rc != 0) return __rc;
__bionic_statfs_to_statvfs(&__tmp, __result);
return 0;
}

#endif

#if __ANDROID_API__ < 21

__BIONIC_SYS_STATVFS_INLINE int statvfs64(const char* __path,
struct statvfs64* __result) {
return statvfs(__path, __BIONIC_CAST(reinterpret_cast, struct statvfs*,
__result));
}

__BIONIC_SYS_STATVFS_INLINE int fstatvfs64(int __fd,
struct statvfs64* __result) {
return fstatvfs(__fd, __BIONIC_CAST(reinterpret_cast, struct statvfs*,
__result));
}

#endif

__END_DECLS

#endif
Loading

0 comments on commit 06c99a3

Please sign in to comment.