/*
* 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