/*
|
* 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>
|
|
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, std::string* loopdev) const {
|
if (!FindFreeLoopDevice(loopdev)) {
|
LOG(ERROR) << "Failed to attach, no free loop devices";
|
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_SET_FD, file_fd);
|
if (rc < 0) {
|
PLOG(ERROR) << "Failed LOOP_SET_FD";
|
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;
|
}
|
|
LoopDevice::LoopDevice(int fd, bool auto_close) : fd_(fd), owns_fd_(auto_close) {
|
Init();
|
}
|
|
LoopDevice::LoopDevice(const std::string& path) : fd_(-1), owns_fd_(true) {
|
fd_.reset(open(path.c_str(), O_RDWR | O_CLOEXEC));
|
if (fd_ < -1) {
|
PLOG(ERROR) << "open failed for " << path;
|
return;
|
}
|
Init();
|
}
|
|
LoopDevice::~LoopDevice() {
|
if (valid()) {
|
control_.Detach(device_);
|
}
|
if (!owns_fd_) {
|
(void)fd_.release();
|
}
|
}
|
|
void LoopDevice::Init() {
|
control_.Attach(fd_, &device_);
|
}
|
|
} // namespace dm
|
} // namespace android
|