blob: 08cf5fe1cc2d16d1af0152913694b70e2a49914c [file] [log] [blame]
//
// Copyright (C) 2010 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.
//
#ifndef UPDATE_ENGINE_PAYLOAD_GENERATOR_EXTENT_RANGES_H_
#define UPDATE_ENGINE_PAYLOAD_GENERATOR_EXTENT_RANGES_H_
#include <map>
#include <set>
#include <vector>
#include <base/macros.h>
#include "update_engine/common/utils.h"
#include "update_engine/update_metadata.pb.h"
// An ExtentRanges object represents an unordered collection of extents (and
// therefore blocks). Such an object may be modified by adding or subtracting
// blocks (think: set addition or set subtraction). Note that ExtentRanges
// ignores sparse hole extents mostly to avoid confusion between extending a
// sparse hole range vs. set addition but also to ensure that the delta
// generator doesn't use sparse holes as scratch space.
namespace chromeos_update_engine {
struct ExtentLess {
bool operator()(const Extent& x, const Extent& y) const {
if (x.start_block() == y.start_block()) {
return x.num_blocks() < y.num_blocks();
}
return x.start_block() < y.start_block();
}
};
Extent ExtentForRange(uint64_t start_block, uint64_t num_blocks);
Extent ExtentForBytes(uint64_t block_size,
uint64_t start_bytes,
uint64_t size_bytes);
class ExtentRanges {
public:
typedef std::set<Extent, ExtentLess> ExtentSet;
ExtentRanges() = default;
// When |merge_touching_extents| is set to false, extents that are only
// touching but not overlapping won't be merged. This slightly decreases
// space/time efficiency, but should not impact correctness.
// Only intended usecase is for VABC XOR.
// E.g. [5-9] and [10-14] will be merged iff |merge_touching_extents| is true
explicit ExtentRanges(bool merge_touching_extents)
: merge_touching_extents_(merge_touching_extents) {}
void AddBlock(uint64_t block);
void SubtractBlock(uint64_t block);
void AddExtent(Extent extent);
void SubtractExtent(const Extent& extent);
void AddExtents(const std::vector<Extent>& extents);
void SubtractExtents(const std::vector<Extent>& extents);
void AddRepeatedExtents(
const ::google::protobuf::RepeatedPtrField<Extent>& exts);
void SubtractRepeatedExtents(
const ::google::protobuf::RepeatedPtrField<Extent>& exts);
void AddRanges(const ExtentRanges& ranges);
void SubtractRanges(const ExtentRanges& ranges);
// Returns true if the input extent overlaps with the current ExtentRanges.
bool OverlapsWithExtent(const Extent& extent) const;
// Returns whether the block |block| is in this ExtentRange.
bool ContainsBlock(uint64_t block) const;
static bool ExtentsOverlapOrTouch(const Extent& a, const Extent& b);
static bool ExtentsOverlap(const Extent& a, const Extent& b);
// Dumps contents to the log file. Useful for debugging.
void Dump() const;
uint64_t blocks() const { return blocks_; }
const ExtentSet& extent_set() const { return extent_set_; }
// Returns an ordered vector of extents for |count| blocks,
// using extents in extent_set_. The returned extents are not
// removed from extent_set_. |count| must be less than or equal to
// the number of blocks in this extent set.
std::vector<Extent> GetExtentsForBlockCount(uint64_t count) const;
// Compute the intersection between this ExtentRange and the |extent|
// parameter. Return results in a vector. If there's no intersection, an empty
// vector is returned.
std::vector<Extent> GetIntersectingExtents(const Extent& extent) const;
// Get a range of extents that possibly intersect with |extent|. (Returned
// extents do not necessarily intersect!). It is perfectly acceptable to just
// return all extents in this set, though more efficient solution using binary
// search is preferred.
Range<ExtentSet::const_iterator> GetCandidateRange(
const Extent& extent) const;
private:
ExtentSet extent_set_;
uint64_t blocks_ = 0;
bool merge_touching_extents_ = true;
};
// Filters out from the passed list of extents |extents| all the blocks in the
// ExtentRanges set. Note that the order of the blocks in |extents| is preserved
// omitting blocks present in the ExtentRanges |ranges|.
std::vector<Extent> FilterExtentRanges(const std::vector<Extent>& extents,
const ExtentRanges& ranges);
Extent GetOverlapExtent(const Extent& extent1, const Extent& extent2);
} // namespace chromeos_update_engine
#endif // UPDATE_ENGINE_PAYLOAD_GENERATOR_EXTENT_RANGES_H_