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