/* * Copyright (C) 2015 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 "F2fs.h" #include "Utils.h" #include #include #include #include #include #include #include #include #include using android::base::StringPrintf; namespace android { namespace vold { namespace f2fs { static const char* kMkfsPath = "/system/bin/make_f2fs"; static const char* kFsckPath = "/system/bin/fsck.f2fs"; bool IsSupported() { return access(kMkfsPath, X_OK) == 0 && access(kFsckPath, X_OK) == 0 && IsFilesystemSupported("f2fs"); } status_t Check(const std::string& source) { std::vector cmd; cmd.push_back(kFsckPath); cmd.push_back("-a"); cmd.push_back(source); // f2fs devices are currently always trusted return ForkExecvp(cmd, nullptr, sFsckContext); } status_t Mount(const std::string& source, const std::string& target) { const char* c_source = source.c_str(); const char* c_target = target.c_str(); unsigned long flags = MS_NOATIME | MS_NODEV | MS_NOSUID | MS_DIRSYNC; int res = mount(c_source, c_target, "f2fs", flags, NULL); if (res != 0) { PLOG(ERROR) << "Failed to mount " << source; if (errno == EROFS) { res = mount(c_source, c_target, "f2fs", flags | MS_RDONLY, NULL); if (res != 0) { PLOG(ERROR) << "Failed to mount read-only " << source; } } } return res; } status_t Format(const std::string& source, bool is_zoned, const std::vector& user_devices, const std::vector& device_aliased, int64_t length) { std::vector cmd; /* '-g android' parameter passed here which defaults the sector size to 4096 */ static constexpr int kSectorSize = 4096; cmd.emplace_back(kMkfsPath); cmd.emplace_back("-f"); cmd.emplace_back("-d1"); cmd.emplace_back("-g"); cmd.emplace_back("android"); if (android::base::GetBoolProperty("vold.has_compress", false)) { cmd.emplace_back("-O"); cmd.emplace_back("compression"); cmd.emplace_back("-O"); cmd.emplace_back("extra_attr"); } const bool needs_casefold = android::base::GetBoolProperty("external_storage.casefold.enabled", false); if (needs_casefold) { cmd.emplace_back("-O"); cmd.emplace_back("casefold"); cmd.emplace_back("-C"); cmd.emplace_back("utf8"); } if (is_zoned) { cmd.emplace_back("-m"); } for (size_t i = 0; i < user_devices.size(); i++) { std::string device_name = user_devices[i]; cmd.emplace_back("-c"); if (device_aliased[i]) { std::filesystem::path path = device_name; device_name += "@" + path.filename().string(); } cmd.emplace_back(device_name); } cmd.emplace_back("-b"); cmd.emplace_back(std::to_string(getpagesize())); cmd.emplace_back(source.c_str()); if (length) { cmd.emplace_back(std::to_string(length / kSectorSize)); } std::vector cmd_cstrs; for (auto& arg : cmd) { cmd_cstrs.emplace_back(arg.c_str()); } return logwrap_fork_execvp(cmd_cstrs.size(), cmd_cstrs.data(), nullptr, false, LOG_KLOG, false, nullptr); } } // namespace f2fs } // namespace vold } // namespace android