xref: /aosp_15_r20/system/apex/apexd/apexd_private.cpp (revision 33f3758387333dbd2962d7edbd98681940d895da)
1*33f37583SAndroid Build Coastguard Worker /*
2*33f37583SAndroid Build Coastguard Worker  * Copyright (C) 2018 The Android Open Source Project
3*33f37583SAndroid Build Coastguard Worker  *
4*33f37583SAndroid Build Coastguard Worker  * Licensed under the Apache License, Version 2.0 (the "License");
5*33f37583SAndroid Build Coastguard Worker  * you may not use this file except in compliance with the License.
6*33f37583SAndroid Build Coastguard Worker  * You may obtain a copy of the License at
7*33f37583SAndroid Build Coastguard Worker  *
8*33f37583SAndroid Build Coastguard Worker  *      http://www.apache.org/licenses/LICENSE-2.0
9*33f37583SAndroid Build Coastguard Worker  *
10*33f37583SAndroid Build Coastguard Worker  * Unless required by applicable law or agreed to in writing, software
11*33f37583SAndroid Build Coastguard Worker  * distributed under the License is distributed on an "AS IS" BASIS,
12*33f37583SAndroid Build Coastguard Worker  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*33f37583SAndroid Build Coastguard Worker  * See the License for the specific language governing permissions and
14*33f37583SAndroid Build Coastguard Worker  * limitations under the License.
15*33f37583SAndroid Build Coastguard Worker  */
16*33f37583SAndroid Build Coastguard Worker 
17*33f37583SAndroid Build Coastguard Worker #include "apexd_private.h"
18*33f37583SAndroid Build Coastguard Worker 
19*33f37583SAndroid Build Coastguard Worker #include <sys/mount.h>
20*33f37583SAndroid Build Coastguard Worker #include <sys/stat.h>
21*33f37583SAndroid Build Coastguard Worker 
22*33f37583SAndroid Build Coastguard Worker #include <android-base/logging.h>
23*33f37583SAndroid Build Coastguard Worker #include <android-base/macros.h>
24*33f37583SAndroid Build Coastguard Worker 
25*33f37583SAndroid Build Coastguard Worker #include "string_log.h"
26*33f37583SAndroid Build Coastguard Worker 
27*33f37583SAndroid Build Coastguard Worker using android::base::ErrnoError;
28*33f37583SAndroid Build Coastguard Worker using android::base::Result;
29*33f37583SAndroid Build Coastguard Worker 
30*33f37583SAndroid Build Coastguard Worker namespace android {
31*33f37583SAndroid Build Coastguard Worker namespace apex {
32*33f37583SAndroid Build Coastguard Worker namespace apexd_private {
33*33f37583SAndroid Build Coastguard Worker 
BindMount(const std::string & target,const std::string & source)34*33f37583SAndroid Build Coastguard Worker Result<void> BindMount(const std::string& target, const std::string& source) {
35*33f37583SAndroid Build Coastguard Worker   LOG(VERBOSE) << "Creating bind-mount for " << target << " for " << source;
36*33f37583SAndroid Build Coastguard Worker   // Ensure the directory exists, try to unmount.
37*33f37583SAndroid Build Coastguard Worker   {
38*33f37583SAndroid Build Coastguard Worker     bool exists;
39*33f37583SAndroid Build Coastguard Worker     bool is_dir;
40*33f37583SAndroid Build Coastguard Worker     {
41*33f37583SAndroid Build Coastguard Worker       struct stat buf;
42*33f37583SAndroid Build Coastguard Worker       if (stat(target.c_str(), &buf) != 0) {
43*33f37583SAndroid Build Coastguard Worker         if (errno == ENOENT) {
44*33f37583SAndroid Build Coastguard Worker           exists = false;
45*33f37583SAndroid Build Coastguard Worker           is_dir = false;
46*33f37583SAndroid Build Coastguard Worker         } else {
47*33f37583SAndroid Build Coastguard Worker           PLOG(ERROR) << "Could not stat target directory " << target;
48*33f37583SAndroid Build Coastguard Worker           // Still attempt to bind-mount.
49*33f37583SAndroid Build Coastguard Worker           exists = true;
50*33f37583SAndroid Build Coastguard Worker           is_dir = true;
51*33f37583SAndroid Build Coastguard Worker         }
52*33f37583SAndroid Build Coastguard Worker       } else {
53*33f37583SAndroid Build Coastguard Worker         exists = true;
54*33f37583SAndroid Build Coastguard Worker         is_dir = S_ISDIR(buf.st_mode);
55*33f37583SAndroid Build Coastguard Worker       }
56*33f37583SAndroid Build Coastguard Worker     }
57*33f37583SAndroid Build Coastguard Worker 
58*33f37583SAndroid Build Coastguard Worker     // Ensure that it is a folder.
59*33f37583SAndroid Build Coastguard Worker     if (exists && !is_dir) {
60*33f37583SAndroid Build Coastguard Worker       LOG(WARNING) << target << " is not a directory, attempting to fix";
61*33f37583SAndroid Build Coastguard Worker       if (unlink(target.c_str()) != 0) {
62*33f37583SAndroid Build Coastguard Worker         PLOG(ERROR) << "Failed to unlink " << target;
63*33f37583SAndroid Build Coastguard Worker         // Try mkdir, anyways.
64*33f37583SAndroid Build Coastguard Worker       }
65*33f37583SAndroid Build Coastguard Worker       exists = false;
66*33f37583SAndroid Build Coastguard Worker     }
67*33f37583SAndroid Build Coastguard Worker     // And create it if necessary.
68*33f37583SAndroid Build Coastguard Worker     if (!exists) {
69*33f37583SAndroid Build Coastguard Worker       LOG(VERBOSE) << "Creating mountpoint " << target;
70*33f37583SAndroid Build Coastguard Worker       if (mkdir(target.c_str(), kMkdirMode) != 0) {
71*33f37583SAndroid Build Coastguard Worker         return ErrnoError() << "Could not create mountpoint " << target;
72*33f37583SAndroid Build Coastguard Worker       }
73*33f37583SAndroid Build Coastguard Worker     };
74*33f37583SAndroid Build Coastguard Worker     // Unmount any active bind-mount.
75*33f37583SAndroid Build Coastguard Worker     if (exists) {
76*33f37583SAndroid Build Coastguard Worker       int rc = umount2(target.c_str(), UMOUNT_NOFOLLOW);
77*33f37583SAndroid Build Coastguard Worker       if (rc != 0 && errno != EINVAL) {
78*33f37583SAndroid Build Coastguard Worker         // Log error but ignore.
79*33f37583SAndroid Build Coastguard Worker         PLOG(ERROR) << "Could not unmount " << target;
80*33f37583SAndroid Build Coastguard Worker       }
81*33f37583SAndroid Build Coastguard Worker     }
82*33f37583SAndroid Build Coastguard Worker   }
83*33f37583SAndroid Build Coastguard Worker 
84*33f37583SAndroid Build Coastguard Worker   LOG(VERBOSE) << "Bind-mounting " << source << " to " << target;
85*33f37583SAndroid Build Coastguard Worker   if (mount(source.c_str(), target.c_str(), nullptr, MS_BIND, nullptr) == 0) {
86*33f37583SAndroid Build Coastguard Worker     return {};
87*33f37583SAndroid Build Coastguard Worker   }
88*33f37583SAndroid Build Coastguard Worker   return ErrnoError() << "Could not bind-mount " << source << " to " << target;
89*33f37583SAndroid Build Coastguard Worker }
90*33f37583SAndroid Build Coastguard Worker 
91*33f37583SAndroid Build Coastguard Worker }  // namespace apexd_private
92*33f37583SAndroid Build Coastguard Worker }  // namespace apex
93*33f37583SAndroid Build Coastguard Worker }  // namespace android
94