blob: bd468a1192724b1b56a65b2fc1ed6cd6b6f5f8a0 [file] [log] [blame]
Alex Deymoaea4c1c2015-08-19 20:24:43 -07001//
2// Copyright (C) 2010 The Android Open Source Project
3//
4// Licensed under the Apache License, Version 2.0 (the "License");
5// you may not use this file except in compliance with the License.
6// You may obtain a copy of the License at
7//
8// http://www.apache.org/licenses/LICENSE-2.0
9//
10// Unless required by applicable law or agreed to in writing, software
11// distributed under the License is distributed on an "AS IS" BASIS,
12// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13// See the License for the specific language governing permissions and
14// limitations under the License.
15//
Andrew de los Reyes5fdae4a2010-10-05 10:47:42 -070016
Alex Deymo1beda782015-06-07 23:01:25 +020017#ifndef UPDATE_ENGINE_PAYLOAD_GENERATOR_EXTENT_RANGES_H_
18#define UPDATE_ENGINE_PAYLOAD_GENERATOR_EXTENT_RANGES_H_
Andrew de los Reyes5fdae4a2010-10-05 10:47:42 -070019
Andrew de los Reyes5fdae4a2010-10-05 10:47:42 -070020#include <set>
21#include <vector>
22
Ben Chan05735a12014-09-03 07:48:22 -070023#include <base/macros.h>
Andrew de los Reyes5fdae4a2010-10-05 10:47:42 -070024
Kelvin Zhangd567c8b2021-07-08 14:10:23 -040025#include "update_engine/common/utils.h"
Kelvin Zhange532af32024-01-10 15:29:02 -080026#include "update_engine/payload_generator/extent_utils.h"
Andrew de los Reyes5fdae4a2010-10-05 10:47:42 -070027#include "update_engine/update_metadata.pb.h"
28
Darin Petkov94817cb2013-05-08 14:33:24 +020029// An ExtentRanges object represents an unordered collection of extents (and
30// therefore blocks). Such an object may be modified by adding or subtracting
31// blocks (think: set addition or set subtraction). Note that ExtentRanges
32// ignores sparse hole extents mostly to avoid confusion between extending a
33// sparse hole range vs. set addition but also to ensure that the delta
34// generator doesn't use sparse holes as scratch space.
Andrew de los Reyes5fdae4a2010-10-05 10:47:42 -070035
36namespace chromeos_update_engine {
37
Andrew de los Reyes5fdae4a2010-10-05 10:47:42 -070038Extent ExtentForRange(uint64_t start_block, uint64_t num_blocks);
Sen Jiang0a582fb2018-06-26 19:27:21 -070039Extent ExtentForBytes(uint64_t block_size,
40 uint64_t start_bytes,
41 uint64_t size_bytes);
Andrew de los Reyes5fdae4a2010-10-05 10:47:42 -070042
43class ExtentRanges {
44 public:
45 typedef std::set<Extent, ExtentLess> ExtentSet;
46
Kelvin Zhangbef99c32021-08-18 09:57:02 -070047 ExtentRanges() = default;
48 // When |merge_touching_extents| is set to false, extents that are only
49 // touching but not overlapping won't be merged. This slightly decreases
50 // space/time efficiency, but should not impact correctness.
51 // Only intended usecase is for VABC XOR.
52 // E.g. [5-9] and [10-14] will be merged iff |merge_touching_extents| is true
53 explicit ExtentRanges(bool merge_touching_extents)
54 : merge_touching_extents_(merge_touching_extents) {}
Andrew de los Reyes5fdae4a2010-10-05 10:47:42 -070055 void AddBlock(uint64_t block);
56 void SubtractBlock(uint64_t block);
57 void AddExtent(Extent extent);
58 void SubtractExtent(const Extent& extent);
59 void AddExtents(const std::vector<Extent>& extents);
60 void SubtractExtents(const std::vector<Extent>& extents);
61 void AddRepeatedExtents(
Amin Hassani232f8f92019-01-14 16:15:31 -080062 const ::google::protobuf::RepeatedPtrField<Extent>& exts);
Andrew de los Reyes5fdae4a2010-10-05 10:47:42 -070063 void SubtractRepeatedExtents(
Amin Hassani232f8f92019-01-14 16:15:31 -080064 const ::google::protobuf::RepeatedPtrField<Extent>& exts);
Andrew de los Reyes5fdae4a2010-10-05 10:47:42 -070065 void AddRanges(const ExtentRanges& ranges);
66 void SubtractRanges(const ExtentRanges& ranges);
67
Tianjie87af6c02020-08-11 15:06:26 -070068 // Returns true if the input extent overlaps with the current ExtentRanges.
69 bool OverlapsWithExtent(const Extent& extent) const;
70
Alex Deymof0061352015-07-01 14:59:15 -070071 // Returns whether the block |block| is in this ExtentRange.
72 bool ContainsBlock(uint64_t block) const;
73
Andrew de los Reyes5fdae4a2010-10-05 10:47:42 -070074 static bool ExtentsOverlapOrTouch(const Extent& a, const Extent& b);
75 static bool ExtentsOverlap(const Extent& a, const Extent& b);
76
77 // Dumps contents to the log file. Useful for debugging.
78 void Dump() const;
Darin Petkov94817cb2013-05-08 14:33:24 +020079
Andrew de los Reyes5fdae4a2010-10-05 10:47:42 -070080 uint64_t blocks() const { return blocks_; }
81 const ExtentSet& extent_set() const { return extent_set_; }
82
83 // Returns an ordered vector of extents for |count| blocks,
84 // using extents in extent_set_. The returned extents are not
85 // removed from extent_set_. |count| must be less than or equal to
86 // the number of blocks in this extent set.
87 std::vector<Extent> GetExtentsForBlockCount(uint64_t count) const;
88
Kelvin Zhangd567c8b2021-07-08 14:10:23 -040089 // Compute the intersection between this ExtentRange and the |extent|
90 // parameter. Return results in a vector. If there's no intersection, an empty
91 // vector is returned.
92 std::vector<Extent> GetIntersectingExtents(const Extent& extent) const;
93
94 // Get a range of extents that possibly intersect with |extent|. (Returned
95 // extents do not necessarily intersect!). It is perfectly acceptable to just
96 // return all extents in this set, though more efficient solution using binary
97 // search is preferred.
98 Range<ExtentSet::const_iterator> GetCandidateRange(
99 const Extent& extent) const;
100
Andrew de los Reyes5fdae4a2010-10-05 10:47:42 -0700101 private:
102 ExtentSet extent_set_;
Kelvin Zhangbef99c32021-08-18 09:57:02 -0700103 uint64_t blocks_ = 0;
Kelvin Zhang76f10b82021-06-25 18:45:46 -0400104 bool merge_touching_extents_ = true;
Andrew de los Reyes5fdae4a2010-10-05 10:47:42 -0700105};
106
Alex Deymoa376a6e2015-06-05 00:31:34 +0200107// Filters out from the passed list of extents |extents| all the blocks in the
108// ExtentRanges set. Note that the order of the blocks in |extents| is preserved
109// omitting blocks present in the ExtentRanges |ranges|.
110std::vector<Extent> FilterExtentRanges(const std::vector<Extent>& extents,
111 const ExtentRanges& ranges);
112
Kelvin Zhangd567c8b2021-07-08 14:10:23 -0400113Extent GetOverlapExtent(const Extent& extent1, const Extent& extent2);
114
Andrew de los Reyes5fdae4a2010-10-05 10:47:42 -0700115} // namespace chromeos_update_engine
116
Alex Deymo1beda782015-06-07 23:01:25 +0200117#endif // UPDATE_ENGINE_PAYLOAD_GENERATOR_EXTENT_RANGES_H_