// Copyright 2016 Benjamin Barenblat // // 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 "posix_extras.h" #include #include #include #include #include #include #include #include #include #include #include #include #include #include namespace scoville { namespace { std::system_error SystemError() { return std::system_error(errno, std::system_category()); } } // namespace File::File(const char* const path, const int flags, const mode_t mode) : path_(path) { if ((fd_ = open(path, flags, mode)) == -1) { throw SystemError(); } VLOG(1) << "opening file descriptor " << fd_; } File::File(const File& other) : path_(other.path_), fd_(other.Duplicate()) { VLOG(1) << "opening file descriptor " << fd_; } File::~File() noexcept { VLOG(1) << "closing file descriptor " << fd_; if (close(fd_) == -1) { LOG(ERROR) << "failed to close file descriptor " << fd_; } } struct stat File::Stat() const { struct stat result; if (fstat(fd_, &result) == -1) { throw SystemError(); } return result; } void File::ChModAt(const char* const path, const mode_t mode) const { if (path[0] == '/') { throw std::invalid_argument("absolute path"); } if (fchmodat(fd_, path, mode, 0) == -1) { throw SystemError(); } } struct stat File::LinkStatAt(const char* const path) const { if (path[0] == '/') { throw std::invalid_argument("absolute path"); } struct stat result; if (fstatat(fd_, path, &result, AT_SYMLINK_NOFOLLOW) == -1) { throw SystemError(); } return result; } void File::MkDir(const char* const path, const mode_t mode) const { if (path[0] == '/') { throw std::invalid_argument("absolute path"); } if (mkdirat(fd_, path, mode | S_IFDIR) == -1) { throw SystemError(); } } void File::MkNod(const char* const path, const mode_t mode, const dev_t dev) const { if (path[0] == '/') { throw std::invalid_argument("absolute path"); } if (mknodat(fd_, path, mode, dev) == -1) { throw SystemError(); } } File File::OpenAt(const char* const path, const int flags, const mode_t mode) const { if (path[0] == '/') { throw std::invalid_argument("absolute path"); } File result; if ((result.fd_ = openat(fd_, path, flags, mode)) == -1) { throw SystemError(); } result.path_ = path_ + "/" + path; return result; } std::vector File::Read(off_t offset, size_t bytes) const { std::vector result(bytes, 0); size_t cursor = 0; ssize_t bytes_read; while (0 < (bytes_read = pread(fd_, result.data() + cursor, bytes, offset))) { cursor += static_cast(bytes_read); offset += bytes_read; bytes -= static_cast(bytes_read); } if (bytes_read == -1) { throw SystemError(); } result.resize(cursor); return result; } void File::RenameAt(const char* old_path, const char* new_path) const { if (old_path[0] == '/' || new_path[0] == '/') { throw std::invalid_argument("absolute path"); } if (renameat(fd_, old_path, fd_, new_path) == -1) { throw SystemError(); } } void File::RmDirAt(const char* const path) const { if (path[0] == '/') { throw std::invalid_argument("absolute path"); } if (unlinkat(fd_, path, AT_REMOVEDIR) == -1) { throw SystemError(); } } void File::UnlinkAt(const char* const path) const { if (path[0] == '/') { throw std::invalid_argument("absolute path"); } if (unlinkat(fd_, path, 0) == -1) { throw SystemError(); } } void File::UTimeNs(const char* const path, const timespec& access, const timespec& modification) const { if (path[0] == '/') { throw std::invalid_argument("absolute path"); } std::array times{{access, modification}}; if (utimensat(fd_, path, times.data(), AT_SYMLINK_NOFOLLOW) == -1) { throw SystemError(); } } size_t File::Write(const off_t offset, const std::vector& to_write) { size_t bytes_written = 0; while (bytes_written < to_write.size()) { const ssize_t pwrite_result = pwrite( fd_, to_write.data() + bytes_written, to_write.size() - bytes_written, offset + static_cast(bytes_written)); if (pwrite_result == -1) { throw SystemError(); } bytes_written += static_cast(pwrite_result); } return bytes_written; } int File::Duplicate() const { int result; if ((result = dup(fd_)) == -1) { throw SystemError(); } return result; } Directory::Directory(const File& file) { // "After a successful call to fdopendir(), fd is used internally by the // implementation, and should not otherwise be used by the application." We // therefore need to grab an unmanaged copy of the file descriptor from file. if (!(stream_ = fdopendir(file.Duplicate()))) { throw SystemError(); } rewinddir(stream_); } Directory::~Directory() noexcept { if (closedir(stream_) == -1) { LOG(ERROR) << "failed to close directory stream"; } } long Directory::offset() const { long result; if ((result = telldir(stream_)) == -1) { throw SystemError(); } return result; } void Directory::Seek(const long offset) noexcept { seekdir(stream_, offset); } std::experimental::optional Directory::ReadOne() { dirent* result; errno = 0; if (!(result = readdir(stream_))) { if (errno == 0) { return std::experimental::nullopt; } else { throw SystemError(); } } return *result; } } // scoville