1 /*
2  * Copyright (C) 2015 The Android Open Source Project
3  *
4  * Licensed under the Apache License, Version 2.0 (the "License");
5  * you may not use this file except in compliance with the License.
6  * You may obtain a copy of the License at
7  *
8  *      http://www.apache.org/licenses/LICENSE-2.0
9  *
10  * Unless required by applicable law or agreed to in writing, software
11  * distributed under the License is distributed on an "AS IS" BASIS,
12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13  * See the License for the specific language governing permissions and
14  * limitations under the License.
15  */
16 
17 #include "F2fs.h"
18 #include "Utils.h"
19 
20 #include <android-base/logging.h>
21 #include <android-base/properties.h>
22 #include <android-base/stringprintf.h>
23 #include <logwrap/logwrap.h>
24 #include <fscrypt/fscrypt.h>
25 
26 #include <string>
27 #include <vector>
28 
29 #include <sys/mount.h>
30 
31 using android::base::StringPrintf;
32 
33 namespace android {
34 namespace vold {
35 namespace f2fs {
36 
37 static const char* kMkfsPath = "/system/bin/make_f2fs";
38 static const char* kFsckPath = "/system/bin/fsck.f2fs";
39 
IsSupported()40 bool IsSupported() {
41     return access(kMkfsPath, X_OK) == 0 && access(kFsckPath, X_OK) == 0 &&
42            IsFilesystemSupported("f2fs");
43 }
44 
Check(const std::string & source)45 status_t Check(const std::string& source) {
46     std::vector<std::string> cmd;
47     cmd.push_back(kFsckPath);
48     cmd.push_back("-a");
49     cmd.push_back(source);
50 
51     // f2fs devices are currently always trusted
52     return ForkExecvp(cmd, nullptr, sFsckContext);
53 }
54 
Mount(const std::string & source,const std::string & target)55 status_t Mount(const std::string& source, const std::string& target) {
56     const char* c_source = source.c_str();
57     const char* c_target = target.c_str();
58     unsigned long flags = MS_NOATIME | MS_NODEV | MS_NOSUID | MS_DIRSYNC;
59 
60     int res = mount(c_source, c_target, "f2fs", flags, NULL);
61     if (res != 0) {
62         PLOG(ERROR) << "Failed to mount " << source;
63         if (errno == EROFS) {
64             res = mount(c_source, c_target, "f2fs", flags | MS_RDONLY, NULL);
65             if (res != 0) {
66                 PLOG(ERROR) << "Failed to mount read-only " << source;
67             }
68         }
69     }
70 
71     return res;
72 }
73 
Format(const std::string & source,bool is_zoned,const std::vector<std::string> & user_devices)74 status_t Format(const std::string& source, bool is_zoned,
75                 const std::vector<std::string>& user_devices) {
76     std::vector<char const*> cmd;
77     cmd.emplace_back(kMkfsPath);
78 
79     cmd.emplace_back("-f");
80     cmd.emplace_back("-d1");
81 
82     cmd.emplace_back("-g");
83     cmd.emplace_back("android");
84 
85     if (android::base::GetBoolProperty("vold.has_compress", false)) {
86         cmd.emplace_back("-O");
87         cmd.emplace_back("compression");
88         cmd.emplace_back("-O");
89         cmd.emplace_back("extra_attr");
90     }
91 
92     const bool needs_casefold =
93             android::base::GetBoolProperty("external_storage.casefold.enabled", false);
94     if (needs_casefold) {
95         cmd.emplace_back("-O");
96         cmd.emplace_back("casefold");
97         cmd.emplace_back("-C");
98         cmd.emplace_back("utf8");
99     }
100     if (is_zoned) {
101         cmd.emplace_back("-m");
102     }
103     for (auto& device : user_devices) {
104         cmd.emplace_back("-c");
105         cmd.emplace_back(device.c_str());
106     }
107     std::string block_size = std::to_string(getpagesize());
108     cmd.emplace_back("-b");
109     cmd.emplace_back(block_size.c_str());
110 
111     cmd.emplace_back(source.c_str());
112 
113     return logwrap_fork_execvp(cmd.size(), cmd.data(), nullptr, false, LOG_KLOG,
114                              false, nullptr);
115 }
116 
117 }  // namespace f2fs
118 }  // namespace vold
119 }  // namespace android
120