blob: bec1025e2207e3994686c0056cf6b79ebc2b5bd4 [file] [log] [blame]
/*
* 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.
*/
#define LOG_TAG "Vold"
#include "EmulatedVolume.h"
#include "Utils.h"
#include <base/stringprintf.h>
#include <cutils/fs.h>
#include <cutils/log.h>
#include <private/android_filesystem_config.h>
#include <fcntl.h>
#include <stdlib.h>
#include <sys/mount.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <sys/wait.h>
using android::base::StringPrintf;
namespace android {
namespace vold {
static const char* kFusePath = "/system/bin/sdcard";
static const char* kUserMountPath = "/mnt/user";
EmulatedVolume::EmulatedVolume(const std::string& rawPath, const std::string& nickname) :
VolumeBase(VolumeType::kEmulated), mFusePid(0), mPrimary(false) {
mRawPath = rawPath;
mFusePath = StringPrintf("/mnt/media_rw/emulated_fuse_%s", nickname.c_str());
}
EmulatedVolume::~EmulatedVolume() {
}
status_t EmulatedVolume::doMount() {
if (fs_prepare_dir(mFusePath.c_str(), 0770, AID_MEDIA_RW, AID_MEDIA_RW)) {
SLOGE("Failed to create mount point %s: %s", mFusePath.c_str(), strerror(errno));
return -errno;
}
if (!(mFusePid = fork())) {
if (execl(kFusePath,
"-u", "1023", // AID_MEDIA_RW
"-g", "1023", // AID_MEDIA_RW
"-d",
mRawPath.c_str(),
mFusePath.c_str())) {
SLOGE("Failed to exec: %s", strerror(errno));
}
_exit(1);
}
if (mFusePid == -1) {
SLOGE("Failed to fork: %s", strerror(errno));
return -errno;
}
return OK;
}
status_t EmulatedVolume::doUnmount() {
if (mFusePid > 0) {
kill(mFusePid, SIGTERM);
TEMP_FAILURE_RETRY(waitpid(mFusePid, nullptr, 0));
mFusePid = 0;
}
ForceUnmount(mFusePath);
TEMP_FAILURE_RETRY(unlink(mFusePath.c_str()));
return OK;
}
status_t EmulatedVolume::doFormat() {
return -ENOTSUP;
}
status_t EmulatedVolume::bindUser(userid_t user) {
return bindUserInternal(user, true);
}
status_t EmulatedVolume::unbindUser(userid_t user) {
return bindUserInternal(user, false);
}
status_t EmulatedVolume::bindUserInternal(userid_t user, bool bind) {
if (!mPrimary) {
// Emulated volumes are only bound when primary
return OK;
}
std::string fromPath(StringPrintf("%s/%ud", mFusePath.c_str(), user));
std::string toPath(StringPrintf("%s/%ud/primary", kUserMountPath, user));
if (bind) {
mountBind(fromPath, toPath);
} else {
unmountBind(toPath);
}
return OK;
}
void EmulatedVolume::setPrimary(bool primary) {
if (getState() != VolumeState::kUnmounted) {
SLOGE("Primary state change requires %s to be unmounted", getId().c_str());
return;
}
mPrimary = primary;
}
} // namespace vold
} // namespace android