Home | History | Annotate | Download | only in payload_generator
      1 //
      2 // Copyright (C) 2015 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 //
     16 
     17 #include "update_engine/payload_generator/inplace_generator.h"
     18 
     19 #include <algorithm>
     20 #include <map>
     21 #include <set>
     22 #include <string>
     23 #include <utility>
     24 #include <vector>
     25 
     26 #include "update_engine/common/utils.h"
     27 #include "update_engine/payload_consumer/payload_constants.h"
     28 #include "update_engine/payload_generator/cycle_breaker.h"
     29 #include "update_engine/payload_generator/delta_diff_generator.h"
     30 #include "update_engine/payload_generator/delta_diff_utils.h"
     31 #include "update_engine/payload_generator/extent_ranges.h"
     32 #include "update_engine/payload_generator/graph_types.h"
     33 #include "update_engine/payload_generator/graph_utils.h"
     34 #include "update_engine/payload_generator/topological_sort.h"
     35 #include "update_engine/update_metadata.pb.h"
     36 
     37 using std::make_pair;
     38 using std::map;
     39 using std::pair;
     40 using std::set;
     41 using std::string;
     42 using std::vector;
     43 
     44 namespace chromeos_update_engine {
     45 
     46 using Block = InplaceGenerator::Block;
     47 
     48 namespace {
     49 
     50 // The only PayloadVersion supported by this implementation.
     51 const PayloadVersion kInPlacePayloadVersion{kChromeOSMajorPayloadVersion,
     52                                             kInPlaceMinorPayloadVersion};
     53 
     54 // This class allocates non-existent temp blocks, starting from
     55 // kTempBlockStart. Other code is responsible for converting these
     56 // temp blocks into real blocks, as the client can't read or write to
     57 // these blocks.
     58 class DummyExtentAllocator {
     59  public:
     60   vector<Extent> Allocate(const uint64_t block_count) {
     61     vector<Extent> ret(1);
     62     ret[0].set_start_block(next_block_);
     63     ret[0].set_num_blocks(block_count);
     64     next_block_ += block_count;
     65     return ret;
     66   }
     67 
     68  private:
     69   uint64_t next_block_{kTempBlockStart};
     70 };
     71 
     72 // Takes a vector of blocks and returns an equivalent vector of Extent
     73 // objects.
     74 vector<Extent> CompressExtents(const vector<uint64_t>& blocks) {
     75   vector<Extent> new_extents;
     76   for (uint64_t block : blocks) {
     77     AppendBlockToExtents(&new_extents, block);
     78   }
     79   return new_extents;
     80 }
     81 
     82 // Helper class to compare two operations by start block of the first Extent in
     83 // their destination extents given the index of the operations in the graph.
     84 class IndexedInstallOperationsDstComparator {
     85  public:
     86   explicit IndexedInstallOperationsDstComparator(Graph* graph)
     87     : graph_(graph) {}
     88 
     89   // Compares the operations in the vertex a and b of graph_.
     90   bool operator()(size_t a, size_t b) const {
     91     return diff_utils::CompareAopsByDestination((*graph_)[a].aop,
     92                                                 (*graph_)[b].aop);
     93   }
     94 
     95  private:
     96   const Graph* const graph_;
     97 };
     98 
     99 }  // namespace
    100 
    101 void InplaceGenerator::CheckGraph(const Graph& graph) {
    102   for (const Vertex& v : graph) {
    103     CHECK(v.aop.op.has_type());
    104   }
    105 }
    106 
    107 void InplaceGenerator::SubstituteBlocks(
    108     Vertex* vertex,
    109     const vector<Extent>& remove_extents,
    110     const vector<Extent>& replace_extents) {
    111   // First, expand out the blocks that op reads from
    112   vector<uint64_t> read_blocks =
    113       ExpandExtents(vertex->aop.op.src_extents());
    114   {
    115     // Expand remove_extents and replace_extents
    116     vector<uint64_t> remove_extents_expanded = ExpandExtents(remove_extents);
    117     vector<uint64_t> replace_extents_expanded = ExpandExtents(replace_extents);
    118     CHECK_EQ(remove_extents_expanded.size(), replace_extents_expanded.size());
    119     map<uint64_t, uint64_t> conversion;
    120     for (vector<uint64_t>::size_type i = 0;
    121          i < replace_extents_expanded.size(); i++) {
    122       conversion[remove_extents_expanded[i]] = replace_extents_expanded[i];
    123     }
    124     ApplyMap(&read_blocks, conversion);
    125     for (auto& edge_prop_pair : vertex->out_edges) {
    126       vector<uint64_t> write_before_deps_expanded = ExpandExtents(
    127           edge_prop_pair.second.write_extents);
    128       ApplyMap(&write_before_deps_expanded, conversion);
    129       edge_prop_pair.second.write_extents =
    130           CompressExtents(write_before_deps_expanded);
    131     }
    132   }
    133   // Convert read_blocks back to extents
    134   vertex->aop.op.clear_src_extents();
    135   vector<Extent> new_extents = CompressExtents(read_blocks);
    136   StoreExtents(new_extents, vertex->aop.op.mutable_src_extents());
    137 }
    138 
    139 bool InplaceGenerator::CutEdges(Graph* graph,
    140                                 const set<Edge>& edges,
    141                                 vector<CutEdgeVertexes>* out_cuts) {
    142   DummyExtentAllocator scratch_allocator;
    143   vector<CutEdgeVertexes> cuts;
    144   cuts.reserve(edges.size());
    145 
    146   uint64_t scratch_blocks_used = 0;
    147   for (const Edge& edge : edges) {
    148     cuts.resize(cuts.size() + 1);
    149     vector<Extent> old_extents =
    150         (*graph)[edge.first].out_edges[edge.second].extents;
    151     // Choose some scratch space
    152     scratch_blocks_used += graph_utils::EdgeWeight(*graph, edge);
    153     cuts.back().tmp_extents =
    154         scratch_allocator.Allocate(graph_utils::EdgeWeight(*graph, edge));
    155     // create vertex to copy original->scratch
    156     cuts.back().new_vertex = graph->size();
    157     graph->emplace_back();
    158     cuts.back().old_src = edge.first;
    159     cuts.back().old_dst = edge.second;
    160 
    161     EdgeProperties& cut_edge_properties =
    162         (*graph)[edge.first].out_edges.find(edge.second)->second;
    163 
    164     // This should never happen, as we should only be cutting edges between
    165     // real file nodes, and write-before relationships are created from
    166     // a real file node to a temp copy node:
    167     CHECK(cut_edge_properties.write_extents.empty())
    168         << "Can't cut edge that has write-before relationship.";
    169 
    170     // make node depend on the copy operation
    171     (*graph)[edge.first].out_edges.insert(make_pair(graph->size() - 1,
    172                                                     cut_edge_properties));
    173 
    174     // Set src/dst extents and other proto variables for copy operation
    175     graph->back().aop.op.set_type(InstallOperation::MOVE);
    176     StoreExtents(cut_edge_properties.extents,
    177                  graph->back().aop.op.mutable_src_extents());
    178     StoreExtents(cuts.back().tmp_extents,
    179                  graph->back().aop.op.mutable_dst_extents());
    180     graph->back().aop.op.set_src_length(
    181         graph_utils::EdgeWeight(*graph, edge) * kBlockSize);
    182     graph->back().aop.op.set_dst_length(graph->back().aop.op.src_length());
    183 
    184     // make the dest node read from the scratch space
    185     SubstituteBlocks(
    186         &((*graph)[edge.second]),
    187         (*graph)[edge.first].out_edges[edge.second].extents,
    188         cuts.back().tmp_extents);
    189 
    190     // delete the old edge
    191     CHECK_EQ(static_cast<Graph::size_type>(1),
    192              (*graph)[edge.first].out_edges.erase(edge.second));
    193 
    194     // Add an edge from dst to copy operation
    195     EdgeProperties write_before_edge_properties;
    196     write_before_edge_properties.write_extents = cuts.back().tmp_extents;
    197     (*graph)[edge.second].out_edges.insert(
    198         make_pair(graph->size() - 1, write_before_edge_properties));
    199   }
    200   out_cuts->swap(cuts);
    201   return true;
    202 }
    203 
    204 // Creates all the edges for the graph. Writers of a block point to
    205 // readers of the same block. This is because for an edge A->B, B
    206 // must complete before A executes.
    207 void InplaceGenerator::CreateEdges(
    208     Graph* graph,
    209     const vector<Block>& blocks) {
    210   for (vector<Block>::size_type i = 0;
    211        i < blocks.size(); i++) {
    212     // Blocks with both a reader and writer get an edge
    213     if (blocks[i].reader == Vertex::kInvalidIndex ||
    214         blocks[i].writer == Vertex::kInvalidIndex)
    215       continue;
    216     // Don't have a node depend on itself
    217     if (blocks[i].reader == blocks[i].writer)
    218       continue;
    219     // See if there's already an edge we can add onto
    220     Vertex::EdgeMap::iterator edge_it =
    221         (*graph)[blocks[i].writer].out_edges.find(blocks[i].reader);
    222     if (edge_it == (*graph)[blocks[i].writer].out_edges.end()) {
    223       // No existing edge. Create one
    224       (*graph)[blocks[i].writer].out_edges.insert(
    225           make_pair(blocks[i].reader, EdgeProperties()));
    226       edge_it = (*graph)[blocks[i].writer].out_edges.find(blocks[i].reader);
    227       CHECK(edge_it != (*graph)[blocks[i].writer].out_edges.end());
    228     }
    229     AppendBlockToExtents(&edge_it->second.extents, i);
    230   }
    231 }
    232 
    233 namespace {
    234 
    235 class SortCutsByTopoOrderLess {
    236  public:
    237   explicit SortCutsByTopoOrderLess(
    238       const vector<vector<Vertex::Index>::size_type>& table)
    239       : table_(table) {}
    240   bool operator()(const CutEdgeVertexes& a, const CutEdgeVertexes& b) {
    241     return table_[a.old_dst] < table_[b.old_dst];
    242   }
    243  private:
    244   const vector<vector<Vertex::Index>::size_type>& table_;
    245 };
    246 
    247 }  // namespace
    248 
    249 void InplaceGenerator::GenerateReverseTopoOrderMap(
    250     const vector<Vertex::Index>& op_indexes,
    251     vector<vector<Vertex::Index>::size_type>* reverse_op_indexes) {
    252   vector<vector<Vertex::Index>::size_type> table(op_indexes.size());
    253   for (vector<Vertex::Index>::size_type i = 0, e = op_indexes.size();
    254        i != e; ++i) {
    255     Vertex::Index node = op_indexes[i];
    256     if (table.size() < (node + 1)) {
    257       table.resize(node + 1);
    258     }
    259     table[node] = i;
    260   }
    261   reverse_op_indexes->swap(table);
    262 }
    263 
    264 void InplaceGenerator::SortCutsByTopoOrder(
    265     const vector<Vertex::Index>& op_indexes,
    266     vector<CutEdgeVertexes>* cuts) {
    267   // first, make a reverse lookup table.
    268   vector<vector<Vertex::Index>::size_type> table;
    269   GenerateReverseTopoOrderMap(op_indexes, &table);
    270   SortCutsByTopoOrderLess less(table);
    271   sort(cuts->begin(), cuts->end(), less);
    272 }
    273 
    274 void InplaceGenerator::MoveAndSortFullOpsToBack(
    275     Graph* graph,
    276     vector<Vertex::Index>* op_indexes) {
    277   vector<Vertex::Index> ret;
    278   vector<Vertex::Index> full_ops;
    279   ret.reserve(op_indexes->size());
    280   for (auto op_index : *op_indexes) {
    281     InstallOperation_Type type = (*graph)[op_index].aop.op.type();
    282     if (type == InstallOperation::REPLACE ||
    283         type == InstallOperation::REPLACE_BZ) {
    284       full_ops.push_back(op_index);
    285     } else {
    286       ret.push_back(op_index);
    287     }
    288   }
    289   LOG(INFO) << "Stats: " << full_ops.size() << " full ops out of "
    290             << (full_ops.size() + ret.size()) << " total ops.";
    291   // Sort full ops according to their dst_extents.
    292   sort(full_ops.begin(), full_ops.end(),
    293        IndexedInstallOperationsDstComparator(graph));
    294   ret.insert(ret.end(), full_ops.begin(), full_ops.end());
    295   op_indexes->swap(ret);
    296 }
    297 
    298 namespace {
    299 
    300 template<typename T>
    301 bool TempBlocksExistInExtents(const T& extents) {
    302   for (int i = 0, e = extents.size(); i < e; ++i) {
    303     Extent extent = GetElement(extents, i);
    304     uint64_t start = extent.start_block();
    305     uint64_t num = extent.num_blocks();
    306     if (start >= kTempBlockStart || (start + num) >= kTempBlockStart) {
    307       LOG(ERROR) << "temp block!";
    308       LOG(ERROR) << "start: " << start << ", num: " << num;
    309       LOG(ERROR) << "kTempBlockStart: " << kTempBlockStart;
    310       LOG(ERROR) << "returning true";
    311       return true;
    312     }
    313     // check for wrap-around, which would be a bug:
    314     CHECK(start <= (start + num));
    315   }
    316   return false;
    317 }
    318 
    319 // Converts the cuts, which must all have the same |old_dst| member,
    320 // to full. It does this by converting the |old_dst| to REPLACE or
    321 // REPLACE_BZ, dropping all incoming edges to |old_dst|, and marking
    322 // all temp nodes invalid.
    323 bool ConvertCutsToFull(
    324     Graph* graph,
    325     const string& new_part,
    326     BlobFileWriter* blob_file,
    327     vector<Vertex::Index>* op_indexes,
    328     vector<vector<Vertex::Index>::size_type>* reverse_op_indexes,
    329     const vector<CutEdgeVertexes>& cuts) {
    330   CHECK(!cuts.empty());
    331   set<Vertex::Index> deleted_nodes;
    332   for (const CutEdgeVertexes& cut : cuts) {
    333     TEST_AND_RETURN_FALSE(InplaceGenerator::ConvertCutToFullOp(
    334         graph,
    335         cut,
    336         new_part,
    337         blob_file));
    338     deleted_nodes.insert(cut.new_vertex);
    339   }
    340   deleted_nodes.insert(cuts[0].old_dst);
    341 
    342   vector<Vertex::Index> new_op_indexes;
    343   new_op_indexes.reserve(op_indexes->size());
    344   for (Vertex::Index vertex_index : *op_indexes) {
    345     if (utils::SetContainsKey(deleted_nodes, vertex_index))
    346       continue;
    347     new_op_indexes.push_back(vertex_index);
    348   }
    349   new_op_indexes.push_back(cuts[0].old_dst);
    350   op_indexes->swap(new_op_indexes);
    351   InplaceGenerator::GenerateReverseTopoOrderMap(*op_indexes,
    352                                                 reverse_op_indexes);
    353   return true;
    354 }
    355 
    356 // Tries to assign temp blocks for a collection of cuts, all of which share
    357 // the same old_dst member. If temp blocks can't be found, old_dst will be
    358 // converted to a REPLACE or REPLACE_BZ operation. Returns true on success,
    359 // which can happen even if blocks are converted to full. Returns false
    360 // on exceptional error cases.
    361 bool AssignBlockForAdjoiningCuts(
    362     Graph* graph,
    363     const string& new_part,
    364     BlobFileWriter* blob_file,
    365     vector<Vertex::Index>* op_indexes,
    366     vector<vector<Vertex::Index>::size_type>* reverse_op_indexes,
    367     const vector<CutEdgeVertexes>& cuts) {
    368   CHECK(!cuts.empty());
    369   const Vertex::Index old_dst = cuts[0].old_dst;
    370   // Calculate # of blocks needed
    371   uint64_t blocks_needed = 0;
    372   vector<uint64_t> cuts_blocks_needed(cuts.size());
    373   for (vector<CutEdgeVertexes>::size_type i = 0; i < cuts.size(); ++i) {
    374     uint64_t cut_blocks_needed = 0;
    375     for (const Extent& extent : cuts[i].tmp_extents) {
    376       cut_blocks_needed += extent.num_blocks();
    377     }
    378     blocks_needed += cut_blocks_needed;
    379     cuts_blocks_needed[i] = cut_blocks_needed;
    380   }
    381 
    382   // Find enough blocks
    383   ExtentRanges scratch_ranges;
    384   // Each block that's supplying temp blocks and the corresponding blocks:
    385   typedef vector<pair<Vertex::Index, ExtentRanges>> SupplierVector;
    386   SupplierVector block_suppliers;
    387   uint64_t scratch_blocks_found = 0;
    388   for (vector<Vertex::Index>::size_type i = (*reverse_op_indexes)[old_dst] + 1,
    389            e = op_indexes->size(); i < e; ++i) {
    390     Vertex::Index test_node = (*op_indexes)[i];
    391     if (!(*graph)[test_node].valid)
    392       continue;
    393     // See if this node has sufficient blocks
    394     ExtentRanges ranges;
    395     ranges.AddRepeatedExtents((*graph)[test_node].aop.op.dst_extents());
    396     ranges.SubtractExtent(ExtentForRange(
    397         kTempBlockStart, kSparseHole - kTempBlockStart));
    398     ranges.SubtractRepeatedExtents((*graph)[test_node].aop.op.src_extents());
    399     // For now, for simplicity, subtract out all blocks in read-before
    400     // dependencies.
    401     for (Vertex::EdgeMap::const_iterator edge_i =
    402              (*graph)[test_node].out_edges.begin(),
    403              edge_e = (*graph)[test_node].out_edges.end();
    404          edge_i != edge_e; ++edge_i) {
    405       ranges.SubtractExtents(edge_i->second.extents);
    406     }
    407 
    408     // Prevent using the block 0 as scratch space due to crbug.com/480751.
    409     if (ranges.ContainsBlock(0)) {
    410       LOG(INFO) << "Removing block 0 from the selected scratch range in vertex "
    411                 << i;
    412       ranges.SubtractBlock(0);
    413     }
    414 
    415     if (ranges.blocks() == 0)
    416       continue;
    417 
    418     if (ranges.blocks() + scratch_blocks_found > blocks_needed) {
    419       // trim down ranges
    420       vector<Extent> new_ranges = ranges.GetExtentsForBlockCount(
    421           blocks_needed - scratch_blocks_found);
    422       ranges = ExtentRanges();
    423       ranges.AddExtents(new_ranges);
    424     }
    425     scratch_ranges.AddRanges(ranges);
    426     block_suppliers.push_back(make_pair(test_node, ranges));
    427     scratch_blocks_found += ranges.blocks();
    428     if (scratch_ranges.blocks() >= blocks_needed)
    429       break;
    430   }
    431   if (scratch_ranges.blocks() < blocks_needed) {
    432     LOG(INFO) << "Unable to find sufficient scratch";
    433     TEST_AND_RETURN_FALSE(ConvertCutsToFull(graph,
    434                                             new_part,
    435                                             blob_file,
    436                                             op_indexes,
    437                                             reverse_op_indexes,
    438                                             cuts));
    439     return true;
    440   }
    441   // Use the scratch we found
    442   TEST_AND_RETURN_FALSE(scratch_ranges.blocks() == scratch_blocks_found);
    443 
    444   // Make all the suppliers depend on this node
    445   for (const auto& index_range_pair : block_suppliers) {
    446     graph_utils::AddReadBeforeDepExtents(
    447         &(*graph)[index_range_pair.first],
    448         old_dst,
    449         index_range_pair.second.GetExtentsForBlockCount(
    450             index_range_pair.second.blocks()));
    451   }
    452 
    453   // Replace temp blocks in each cut
    454   for (vector<CutEdgeVertexes>::size_type i = 0; i < cuts.size(); ++i) {
    455     const CutEdgeVertexes& cut = cuts[i];
    456     vector<Extent> real_extents =
    457         scratch_ranges.GetExtentsForBlockCount(cuts_blocks_needed[i]);
    458     scratch_ranges.SubtractExtents(real_extents);
    459 
    460     // Fix the old dest node w/ the real blocks
    461     InplaceGenerator::SubstituteBlocks(&(*graph)[old_dst],
    462                                          cut.tmp_extents,
    463                                          real_extents);
    464 
    465     // Fix the new node w/ the real blocks. Since the new node is just a
    466     // copy operation, we can replace all the dest extents w/ the real
    467     // blocks.
    468     InstallOperation* op = &(*graph)[cut.new_vertex].aop.op;
    469     op->clear_dst_extents();
    470     StoreExtents(real_extents, op->mutable_dst_extents());
    471   }
    472   return true;
    473 }
    474 
    475 }  // namespace
    476 
    477 bool InplaceGenerator::AssignTempBlocks(
    478     Graph* graph,
    479     const string& new_part,
    480     BlobFileWriter* blob_file,
    481     vector<Vertex::Index>* op_indexes,
    482     vector<vector<Vertex::Index>::size_type>* reverse_op_indexes,
    483     const vector<CutEdgeVertexes>& cuts) {
    484   CHECK(!cuts.empty());
    485 
    486   // group of cuts w/ the same old_dst:
    487   vector<CutEdgeVertexes> cuts_group;
    488 
    489   for (vector<CutEdgeVertexes>::size_type i = cuts.size() - 1, e = 0;
    490        true ; --i) {
    491     LOG(INFO) << "Fixing temp blocks in cut " << i
    492               << ": old dst: " << cuts[i].old_dst << " new vertex: "
    493               << cuts[i].new_vertex << " path: "
    494               << (*graph)[cuts[i].old_dst].aop.name;
    495 
    496     if (cuts_group.empty() || (cuts_group[0].old_dst == cuts[i].old_dst)) {
    497       cuts_group.push_back(cuts[i]);
    498     } else {
    499       CHECK(!cuts_group.empty());
    500       TEST_AND_RETURN_FALSE(AssignBlockForAdjoiningCuts(graph,
    501                                                         new_part,
    502                                                         blob_file,
    503                                                         op_indexes,
    504                                                         reverse_op_indexes,
    505                                                         cuts_group));
    506       cuts_group.clear();
    507       cuts_group.push_back(cuts[i]);
    508     }
    509 
    510     if (i == e) {
    511       // break out of for() loop
    512       break;
    513     }
    514   }
    515   CHECK(!cuts_group.empty());
    516   TEST_AND_RETURN_FALSE(AssignBlockForAdjoiningCuts(graph,
    517                                                     new_part,
    518                                                     blob_file,
    519                                                     op_indexes,
    520                                                     reverse_op_indexes,
    521                                                     cuts_group));
    522   return true;
    523 }
    524 
    525 bool InplaceGenerator::NoTempBlocksRemain(const Graph& graph) {
    526   size_t idx = 0;
    527   for (Graph::const_iterator it = graph.begin(), e = graph.end(); it != e;
    528        ++it, ++idx) {
    529     if (!it->valid)
    530       continue;
    531     const InstallOperation& op = it->aop.op;
    532     if (TempBlocksExistInExtents(op.dst_extents()) ||
    533         TempBlocksExistInExtents(op.src_extents())) {
    534       LOG(INFO) << "bad extents in node " << idx;
    535       LOG(INFO) << "so yeah";
    536       return false;
    537     }
    538 
    539     // Check out-edges:
    540     for (const auto& edge_prop_pair : it->out_edges) {
    541       if (TempBlocksExistInExtents(edge_prop_pair.second.extents) ||
    542           TempBlocksExistInExtents(edge_prop_pair.second.write_extents)) {
    543         LOG(INFO) << "bad out edge in node " << idx;
    544         LOG(INFO) << "so yeah";
    545         return false;
    546       }
    547     }
    548   }
    549   return true;
    550 }
    551 
    552 bool InplaceGenerator::ConvertCutToFullOp(Graph* graph,
    553                                           const CutEdgeVertexes& cut,
    554                                           const string& new_part,
    555                                           BlobFileWriter* blob_file) {
    556   // Drop all incoming edges, keep all outgoing edges
    557 
    558   // Keep all outgoing edges
    559   if ((*graph)[cut.old_dst].aop.op.type() != InstallOperation::REPLACE_BZ &&
    560       (*graph)[cut.old_dst].aop.op.type() != InstallOperation::REPLACE) {
    561     Vertex::EdgeMap out_edges = (*graph)[cut.old_dst].out_edges;
    562     graph_utils::DropWriteBeforeDeps(&out_edges);
    563 
    564     // Replace the operation with a REPLACE or REPLACE_BZ to generate the same
    565     // |new_extents| list of blocks and update the graph.
    566     vector<AnnotatedOperation> new_aop;
    567     vector<Extent> new_extents;
    568     ExtentsToVector((*graph)[cut.old_dst].aop.op.dst_extents(),
    569                     &new_extents);
    570     TEST_AND_RETURN_FALSE(diff_utils::DeltaReadFile(
    571         &new_aop,
    572         "",  // old_part
    573         new_part,
    574         vector<Extent>(),  // old_extents
    575         new_extents,
    576         (*graph)[cut.old_dst].aop.name,
    577         -1,  // chunk_blocks, forces to have a single operation.
    578         kInPlacePayloadVersion,
    579         blob_file));
    580     TEST_AND_RETURN_FALSE(new_aop.size() == 1);
    581     TEST_AND_RETURN_FALSE(AddInstallOpToGraph(
    582       graph, cut.old_dst, nullptr, new_aop.front().op, new_aop.front().name));
    583 
    584     (*graph)[cut.old_dst].out_edges = out_edges;
    585 
    586     // Right now we don't have doubly-linked edges, so we have to scan
    587     // the whole graph.
    588     graph_utils::DropIncomingEdgesTo(graph, cut.old_dst);
    589   }
    590 
    591   // Delete temp node
    592   (*graph)[cut.old_src].out_edges.erase(cut.new_vertex);
    593   CHECK((*graph)[cut.old_dst].out_edges.find(cut.new_vertex) ==
    594         (*graph)[cut.old_dst].out_edges.end());
    595   (*graph)[cut.new_vertex].valid = false;
    596   LOG(INFO) << "marked node invalid: " << cut.new_vertex;
    597   return true;
    598 }
    599 
    600 bool InplaceGenerator::ConvertGraphToDag(Graph* graph,
    601                                          const string& new_part,
    602                                          BlobFileWriter* blob_file,
    603                                          vector<Vertex::Index>* final_order,
    604                                          Vertex::Index scratch_vertex) {
    605   CycleBreaker cycle_breaker;
    606   LOG(INFO) << "Finding cycles...";
    607   set<Edge> cut_edges;
    608   cycle_breaker.BreakCycles(*graph, &cut_edges);
    609   LOG(INFO) << "done finding cycles";
    610   CheckGraph(*graph);
    611 
    612   // Calculate number of scratch blocks needed
    613 
    614   LOG(INFO) << "Cutting cycles...";
    615   vector<CutEdgeVertexes> cuts;
    616   TEST_AND_RETURN_FALSE(CutEdges(graph, cut_edges, &cuts));
    617   LOG(INFO) << "done cutting cycles";
    618   LOG(INFO) << "There are " << cuts.size() << " cuts.";
    619   CheckGraph(*graph);
    620 
    621   LOG(INFO) << "Creating initial topological order...";
    622   TopologicalSort(*graph, final_order);
    623   LOG(INFO) << "done with initial topo order";
    624   CheckGraph(*graph);
    625 
    626   LOG(INFO) << "Moving full ops to the back";
    627   MoveAndSortFullOpsToBack(graph, final_order);
    628   LOG(INFO) << "done moving full ops to back";
    629 
    630   vector<vector<Vertex::Index>::size_type> inverse_final_order;
    631   GenerateReverseTopoOrderMap(*final_order, &inverse_final_order);
    632 
    633   SortCutsByTopoOrder(*final_order, &cuts);
    634 
    635   if (!cuts.empty())
    636     TEST_AND_RETURN_FALSE(AssignTempBlocks(graph,
    637                                            new_part,
    638                                            blob_file,
    639                                            final_order,
    640                                            &inverse_final_order,
    641                                            cuts));
    642   LOG(INFO) << "Making sure all temp blocks have been allocated";
    643 
    644   // Remove the scratch node, if any
    645   if (scratch_vertex != Vertex::kInvalidIndex) {
    646     final_order->erase(final_order->begin() +
    647                        inverse_final_order[scratch_vertex]);
    648     (*graph)[scratch_vertex].valid = false;
    649     GenerateReverseTopoOrderMap(*final_order, &inverse_final_order);
    650   }
    651 
    652   graph_utils::DumpGraph(*graph);
    653   CHECK(NoTempBlocksRemain(*graph));
    654   LOG(INFO) << "done making sure all temp blocks are allocated";
    655   return true;
    656 }
    657 
    658 void InplaceGenerator::CreateScratchNode(uint64_t start_block,
    659                                          uint64_t num_blocks,
    660                                          Vertex* vertex) {
    661   vertex->aop.name = "<scratch>";
    662   vertex->aop.op.set_type(InstallOperation::REPLACE_BZ);
    663   vertex->aop.op.set_data_offset(0);
    664   vertex->aop.op.set_data_length(0);
    665   Extent* extent = vertex->aop.op.add_dst_extents();
    666   extent->set_start_block(start_block);
    667   extent->set_num_blocks(num_blocks);
    668 }
    669 
    670 bool InplaceGenerator::AddInstallOpToBlocksVector(
    671     const InstallOperation& operation,
    672     const Graph& graph,
    673     Vertex::Index vertex,
    674     vector<Block>* blocks) {
    675   // See if this is already present.
    676   TEST_AND_RETURN_FALSE(operation.dst_extents_size() > 0);
    677 
    678   enum BlockField { READER = 0, WRITER, BLOCK_FIELD_COUNT };
    679   for (int field = READER; field < BLOCK_FIELD_COUNT; field++) {
    680     const char* past_participle = (field == READER) ? "read" : "written";
    681     const google::protobuf::RepeatedPtrField<Extent>& extents =
    682         (field == READER) ? operation.src_extents() : operation.dst_extents();
    683     Vertex::Index Block::*access_type = (field == READER) ?
    684         &Block::reader : &Block::writer;
    685 
    686     for (const Extent& extent : extents) {
    687       for (uint64_t block = extent.start_block();
    688            block < (extent.start_block() + extent.num_blocks()); block++) {
    689         if ((*blocks)[block].*access_type != Vertex::kInvalidIndex) {
    690           LOG(FATAL) << "Block " << block << " is already "
    691                      << past_participle << " by "
    692                      << (*blocks)[block].*access_type << "("
    693                      << graph[(*blocks)[block].*access_type].aop.name
    694                      << ") and also " << vertex << "("
    695                      << graph[vertex].aop.name << ")";
    696         }
    697         (*blocks)[block].*access_type = vertex;
    698       }
    699     }
    700   }
    701   return true;
    702 }
    703 
    704 bool InplaceGenerator::AddInstallOpToGraph(Graph* graph,
    705                                            Vertex::Index existing_vertex,
    706                                            vector<Block>* blocks,
    707                                            const InstallOperation operation,
    708                                            const string& op_name) {
    709   Vertex::Index vertex = existing_vertex;
    710   if (vertex == Vertex::kInvalidIndex) {
    711     graph->emplace_back();
    712     vertex = graph->size() - 1;
    713   }
    714   (*graph)[vertex].aop.op = operation;
    715   CHECK((*graph)[vertex].aop.op.has_type());
    716   (*graph)[vertex].aop.name = op_name;
    717 
    718   if (blocks)
    719     TEST_AND_RETURN_FALSE(InplaceGenerator::AddInstallOpToBlocksVector(
    720         (*graph)[vertex].aop.op,
    721         *graph,
    722         vertex,
    723         blocks));
    724   return true;
    725 }
    726 
    727 void InplaceGenerator::ApplyMap(vector<uint64_t>* collection,
    728                                 const map<uint64_t, uint64_t>& the_map) {
    729   for (uint64_t& elem : *collection) {
    730     const auto& map_it = the_map.find(elem);
    731     if (map_it != the_map.end())
    732       elem = map_it->second;
    733   }
    734 }
    735 
    736 bool InplaceGenerator::ResolveReadAfterWriteDependencies(
    737     const PartitionConfig& old_part,
    738     const PartitionConfig& new_part,
    739     uint64_t partition_size,
    740     size_t block_size,
    741     BlobFileWriter* blob_file,
    742     vector<AnnotatedOperation>* aops) {
    743   // Convert the operations to the graph.
    744   Graph graph;
    745   CheckGraph(graph);
    746   vector<Block> blocks(std::max(old_part.size, new_part.size) / block_size);
    747   for (const auto& aop : *aops) {
    748     AddInstallOpToGraph(
    749         &graph, Vertex::kInvalidIndex, &blocks, aop.op, aop.name);
    750   }
    751   CheckGraph(graph);
    752 
    753   // Final scratch block (if there's space)
    754   Vertex::Index scratch_vertex = Vertex::kInvalidIndex;
    755   if (blocks.size() < (partition_size / block_size)) {
    756     scratch_vertex = graph.size();
    757     graph.emplace_back();
    758     size_t scratch_blocks = (partition_size / block_size) - blocks.size();
    759     LOG(INFO) << "Added " << scratch_blocks << " scratch space blocks.";
    760     CreateScratchNode(blocks.size(), scratch_blocks, &graph.back());
    761   }
    762   CheckGraph(graph);
    763 
    764   LOG(INFO) << "Creating edges...";
    765   CreateEdges(&graph, blocks);
    766   LOG(INFO) << "Done creating edges";
    767   CheckGraph(graph);
    768 
    769   vector<Vertex::Index> final_order;
    770   TEST_AND_RETURN_FALSE(ConvertGraphToDag(
    771       &graph,
    772       new_part.path,
    773       blob_file,
    774       &final_order,
    775       scratch_vertex));
    776 
    777   // Copy operations over to the |aops| vector in the final_order generated by
    778   // the topological sort.
    779   aops->clear();
    780   aops->reserve(final_order.size());
    781   for (const Vertex::Index vertex_index : final_order) {
    782     const Vertex& vertex = graph[vertex_index];
    783     aops->push_back(vertex.aop);
    784   }
    785   return true;
    786 }
    787 
    788 bool InplaceGenerator::GenerateOperations(
    789     const PayloadGenerationConfig& config,
    790     const PartitionConfig& old_part,
    791     const PartitionConfig& new_part,
    792     BlobFileWriter* blob_file,
    793     vector<AnnotatedOperation>* aops) {
    794   TEST_AND_RETURN_FALSE(old_part.name == new_part.name);
    795   TEST_AND_RETURN_FALSE(config.version.major == kInPlacePayloadVersion.major);
    796   TEST_AND_RETURN_FALSE(config.version.minor == kInPlacePayloadVersion.minor);
    797 
    798   ssize_t hard_chunk_blocks = (config.hard_chunk_size == -1 ? -1 :
    799                                config.hard_chunk_size / config.block_size);
    800   size_t soft_chunk_blocks = config.soft_chunk_size / config.block_size;
    801   uint64_t partition_size = new_part.size;
    802   if (new_part.name == kLegacyPartitionNameRoot)
    803     partition_size = config.rootfs_partition_size;
    804 
    805   LOG(INFO) << "Delta compressing " << new_part.name << " partition...";
    806   TEST_AND_RETURN_FALSE(diff_utils::DeltaReadPartition(aops,
    807                                                        old_part,
    808                                                        new_part,
    809                                                        hard_chunk_blocks,
    810                                                        soft_chunk_blocks,
    811                                                        config.version,
    812                                                        blob_file));
    813   LOG(INFO) << "Done reading " << new_part.name;
    814 
    815   TEST_AND_RETURN_FALSE(ResolveReadAfterWriteDependencies(
    816       old_part, new_part, partition_size, config.block_size, blob_file, aops));
    817   LOG(INFO) << "Done reordering " << new_part.name;
    818   return true;
    819 }
    820 
    821 };  // namespace chromeos_update_engine
    822