178 lines
5.3 KiB
C++
178 lines
5.3 KiB
C++
/*
|
|
* Copyright (C) 2018 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 "libdm/loop_control.h"
|
|
|
|
#include <fcntl.h>
|
|
#include <linux/loop.h>
|
|
#include <stdint.h>
|
|
#include <sys/ioctl.h>
|
|
#include <sys/types.h>
|
|
#include <unistd.h>
|
|
|
|
#include <android-base/logging.h>
|
|
#include <android-base/stringprintf.h>
|
|
#include <android-base/unique_fd.h>
|
|
|
|
#include "utility.h"
|
|
|
|
namespace android {
|
|
namespace dm {
|
|
|
|
LoopControl::LoopControl() : control_fd_(-1) {
|
|
control_fd_.reset(TEMP_FAILURE_RETRY(open(kLoopControlDevice, O_RDWR | O_CLOEXEC)));
|
|
if (control_fd_ < 0) {
|
|
PLOG(ERROR) << "Failed to open loop-control";
|
|
}
|
|
}
|
|
|
|
bool LoopControl::Attach(int file_fd, const std::chrono::milliseconds& timeout_ms,
|
|
std::string* loopdev) const {
|
|
auto start_time = std::chrono::steady_clock::now();
|
|
auto condition = [&]() -> WaitResult {
|
|
if (!FindFreeLoopDevice(loopdev)) {
|
|
LOG(ERROR) << "Failed to attach, no free loop devices";
|
|
return WaitResult::Fail;
|
|
}
|
|
|
|
auto now = std::chrono::steady_clock::now();
|
|
auto time_elapsed = std::chrono::duration_cast<std::chrono::milliseconds>(now - start_time);
|
|
if (!WaitForFile(*loopdev, timeout_ms - time_elapsed)) {
|
|
LOG(ERROR) << "Timed out waiting for path: " << *loopdev;
|
|
return WaitResult::Fail;
|
|
}
|
|
|
|
android::base::unique_fd loop_fd(
|
|
TEMP_FAILURE_RETRY(open(loopdev->c_str(), O_RDWR | O_CLOEXEC)));
|
|
if (loop_fd < 0) {
|
|
PLOG(ERROR) << "Failed to open: " << *loopdev;
|
|
return WaitResult::Fail;
|
|
}
|
|
|
|
if (int rc = ioctl(loop_fd, LOOP_SET_FD, file_fd); rc == 0) {
|
|
return WaitResult::Done;
|
|
}
|
|
if (errno != EBUSY) {
|
|
PLOG(ERROR) << "Failed LOOP_SET_FD";
|
|
return WaitResult::Fail;
|
|
}
|
|
return WaitResult::Wait;
|
|
};
|
|
if (!WaitForCondition(condition, timeout_ms)) {
|
|
LOG(ERROR) << "Timed out trying to acquire a loop device";
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool LoopControl::Detach(const std::string& loopdev) const {
|
|
if (loopdev.empty()) {
|
|
LOG(ERROR) << "Must provide a loop device";
|
|
return false;
|
|
}
|
|
|
|
android::base::unique_fd loop_fd(TEMP_FAILURE_RETRY(open(loopdev.c_str(), O_RDWR | O_CLOEXEC)));
|
|
if (loop_fd < 0) {
|
|
PLOG(ERROR) << "Failed to open: " << loopdev;
|
|
return false;
|
|
}
|
|
|
|
int rc = ioctl(loop_fd, LOOP_CLR_FD, 0);
|
|
if (rc) {
|
|
PLOG(ERROR) << "Failed LOOP_CLR_FD for '" << loopdev << "'";
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool LoopControl::FindFreeLoopDevice(std::string* loopdev) const {
|
|
int rc = ioctl(control_fd_, LOOP_CTL_GET_FREE);
|
|
if (rc < 0) {
|
|
PLOG(ERROR) << "Failed to get free loop device";
|
|
return false;
|
|
}
|
|
|
|
// Ueventd on android creates all loop devices as /dev/block/loopX
|
|
// The total number of available devices is determined by 'loop.max_part'
|
|
// kernel command line argument.
|
|
*loopdev = ::android::base::StringPrintf("/dev/block/loop%d", rc);
|
|
return true;
|
|
}
|
|
|
|
bool LoopControl::EnableDirectIo(int fd) {
|
|
#if !defined(LOOP_SET_BLOCK_SIZE)
|
|
static constexpr int LOOP_SET_BLOCK_SIZE = 0x4C09;
|
|
#endif
|
|
#if !defined(LOOP_SET_DIRECT_IO)
|
|
static constexpr int LOOP_SET_DIRECT_IO = 0x4C08;
|
|
#endif
|
|
|
|
// Note: the block size has to be >= the logical block size of the underlying
|
|
// block device, *not* the filesystem block size.
|
|
if (ioctl(fd, LOOP_SET_BLOCK_SIZE, 4096)) {
|
|
PLOG(ERROR) << "Could not set loop device block size";
|
|
return false;
|
|
}
|
|
if (ioctl(fd, LOOP_SET_DIRECT_IO, 1)) {
|
|
PLOG(ERROR) << "Could not set loop direct IO";
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool LoopControl::SetAutoClearStatus(int fd) {
|
|
struct loop_info64 info = {};
|
|
|
|
info.lo_flags |= LO_FLAGS_AUTOCLEAR;
|
|
if (ioctl(fd, LOOP_SET_STATUS64, &info)) {
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
LoopDevice::LoopDevice(android::base::borrowed_fd fd, const std::chrono::milliseconds& timeout_ms,
|
|
bool auto_close)
|
|
: fd_(fd), owned_fd_(-1) {
|
|
if (auto_close) {
|
|
owned_fd_.reset(fd.get());
|
|
}
|
|
Init(timeout_ms);
|
|
}
|
|
|
|
LoopDevice::LoopDevice(const std::string& path, const std::chrono::milliseconds& timeout_ms)
|
|
: fd_(-1), owned_fd_(-1) {
|
|
owned_fd_.reset(open(path.c_str(), O_RDWR | O_CLOEXEC));
|
|
if (owned_fd_ == -1) {
|
|
PLOG(ERROR) << "open failed for " << path;
|
|
return;
|
|
}
|
|
fd_ = owned_fd_;
|
|
Init(timeout_ms);
|
|
}
|
|
|
|
LoopDevice::~LoopDevice() {
|
|
if (valid()) {
|
|
control_.Detach(device_);
|
|
}
|
|
}
|
|
|
|
void LoopDevice::Init(const std::chrono::milliseconds& timeout_ms) {
|
|
valid_ = control_.Attach(fd_.get(), timeout_ms, &device_);
|
|
}
|
|
|
|
} // namespace dm
|
|
} // namespace android
|