| /* |
| * Copyright (C) 2009 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 "base/logging.h" |
| #include "base/unix_file/fd_file.h" |
| #include <errno.h> |
| #include <sys/stat.h> |
| #include <sys/types.h> |
| #include <unistd.h> |
| |
| namespace unix_file { |
| |
| FdFile::FdFile() : fd_(-1), auto_close_(true) { |
| } |
| |
| FdFile::FdFile(int fd) : fd_(fd), auto_close_(true) { |
| } |
| |
| FdFile::FdFile(int fd, const std::string& path) : fd_(fd), file_path_(path), auto_close_(true) { |
| CHECK_NE(0U, path.size()); |
| } |
| |
| FdFile::~FdFile() { |
| if (auto_close_ && fd_ != -1) { |
| Close(); |
| } |
| } |
| |
| void FdFile::DisableAutoClose() { |
| auto_close_ = false; |
| } |
| |
| bool FdFile::Open(const std::string& path, int flags) { |
| return Open(path, flags, 0640); |
| } |
| |
| bool FdFile::Open(const std::string& path, int flags, mode_t mode) { |
| CHECK_EQ(fd_, -1) << path; |
| fd_ = TEMP_FAILURE_RETRY(open(path.c_str(), flags, mode)); |
| if (fd_ == -1) { |
| return false; |
| } |
| file_path_ = path; |
| return true; |
| } |
| |
| int FdFile::Close() { |
| int result = TEMP_FAILURE_RETRY(close(fd_)); |
| if (result == -1) { |
| return -errno; |
| } else { |
| fd_ = -1; |
| file_path_ = ""; |
| return 0; |
| } |
| } |
| |
| int FdFile::Flush() { |
| int rc = TEMP_FAILURE_RETRY(fdatasync(fd_)); |
| return (rc == -1) ? -errno : rc; |
| } |
| |
| int64_t FdFile::Read(char* buf, int64_t byte_count, int64_t offset) const { |
| int rc = TEMP_FAILURE_RETRY(pread64(fd_, buf, byte_count, offset)); |
| return (rc == -1) ? -errno : rc; |
| } |
| |
| int FdFile::SetLength(int64_t new_length) { |
| int rc = TEMP_FAILURE_RETRY(ftruncate64(fd_, new_length)); |
| return (rc == -1) ? -errno : rc; |
| } |
| |
| int64_t FdFile::GetLength() const { |
| struct stat s; |
| int rc = TEMP_FAILURE_RETRY(fstat(fd_, &s)); |
| return (rc == -1) ? -errno : s.st_size; |
| } |
| |
| int64_t FdFile::Write(const char* buf, int64_t byte_count, int64_t offset) { |
| int rc = TEMP_FAILURE_RETRY(pwrite64(fd_, buf, byte_count, offset)); |
| return (rc == -1) ? -errno : rc; |
| } |
| |
| int FdFile::Fd() const { |
| return fd_; |
| } |
| |
| bool FdFile::IsOpened() const { |
| return fd_ >= 0; |
| } |
| |
| bool FdFile::ReadFully(void* buffer, int64_t byte_count) { |
| char* ptr = static_cast<char*>(buffer); |
| while (byte_count > 0) { |
| int bytes_read = TEMP_FAILURE_RETRY(read(fd_, ptr, byte_count)); |
| if (bytes_read <= 0) { |
| return false; |
| } |
| byte_count -= bytes_read; // Reduce the number of remaining bytes. |
| ptr += bytes_read; // Move the buffer forward. |
| } |
| return true; |
| } |
| |
| bool FdFile::WriteFully(const void* buffer, int64_t byte_count) { |
| const char* ptr = static_cast<const char*>(buffer); |
| while (byte_count > 0) { |
| int bytes_read = TEMP_FAILURE_RETRY(write(fd_, ptr, byte_count)); |
| if (bytes_read < 0) { |
| return false; |
| } |
| byte_count -= bytes_read; // Reduce the number of remaining bytes. |
| ptr += bytes_read; // Move the buffer forward. |
| } |
| return true; |
| } |
| |
| } // namespace unix_file |