// Copyright 2015 The Chromium OS Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "bsdiff/extents_file.h" #include <string.h> #include <algorithm> // Extent files implementation extending FileInterface. // // This class allows to map linear positions in a file to a list of regions in // another file. All the reads and writes are unbuffered, passed directly to the // underlying file. Seeking is done in O(log(N)), where N is the number of // extents in the file, but sequential reads jump to the next extent in O(1). namespace bsdiff { ExtentsFile::ExtentsFile(std::unique_ptr<FileInterface> file, const std::vector<ex_t>& extents) : file_(std::move(file)), extents_(extents) { acc_len_.reserve(extents.size()); for (const ex_t& extent : extents) { acc_len_.push_back(total_ex_len_); total_ex_len_ += extent.len; } } ExtentsFile::~ExtentsFile() { Close(); } bool ExtentsFile::Read(void* buf, size_t count, size_t* bytes_read) { return IOOperation(&FileInterface::Read, buf, count, bytes_read); } bool ExtentsFile::Write(const void* buf, size_t count, size_t* bytes_written) { return IOOperation(&FileInterface::Write, buf, count, bytes_written); } bool ExtentsFile::Seek(off_t pos) { if (pos < 0 || static_cast<uint64_t>(pos) > total_ex_len_) return false; if (acc_len_.empty()) return true; // Note that the first element of acc_len_ is always 0, and pos is at least 0, // so the upper_bound will never return acc_len_.begin(). curr_pos_ = pos; curr_ex_idx_ = std::upper_bound(acc_len_.begin(), acc_len_.end(), pos) - acc_len_.begin(); // We handle the corner case where |pos| is the size of all the extents by // leaving the value of curr_ex_idx_ the same way AdvancePos(0) would leave it // after the seek. if (curr_pos_ < total_ex_len_) curr_ex_idx_--; return true; } bool ExtentsFile::Close() { return file_->Close(); } bool ExtentsFile::GetSize(uint64_t* size) { *size = total_ex_len_; return true; } void ExtentsFile::AdvancePos(uint64_t size) { curr_pos_ += size; for (; curr_ex_idx_ < extents_.size(); curr_ex_idx_++) { if (curr_pos_ < acc_len_[curr_ex_idx_] + extents_[curr_ex_idx_].len) return; } } template <typename T> bool ExtentsFile::IOOperation(bool (FileInterface::*io_op)(T*, size_t, size_t*), T* buf, size_t count, size_t* bytes_processed) { bool result = true; size_t processed = 0; AdvancePos(0); while (count > 0 && curr_ex_idx_ < extents_.size()) { const ex_t& ex = extents_[curr_ex_idx_]; off_t curr_ex_off = curr_pos_ - acc_len_[curr_ex_idx_]; size_t chunk_size = std::min(static_cast<uint64_t>(count), ex.len - curr_ex_off); size_t chunk_processed = 0; if (ex.off < 0) { chunk_processed = chunk_size; } else { if (!file_->Seek(ex.off + curr_ex_off) || !(file_.get()->*io_op)(buf, chunk_size, &chunk_processed)) { processed += chunk_processed; result = processed > 0; break; } } processed += chunk_processed; count -= chunk_processed; // T can be either const void* or void*. We const_cast it back to void* and // then char* to do the arithmetic operation, but |buf| will continue to be // const void* if it was defined that way. buf = static_cast<char*>(const_cast<void*>(buf)) + chunk_processed; AdvancePos(chunk_processed); if (!chunk_processed) break; } *bytes_processed = processed; return result; } } // namespace bsdiff