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:
Allocate(const uint64_t block_count)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.
CompressExtents(const vector<uint64_t> & blocks)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:
IndexedInstallOperationsDstComparator(Graph * graph)86   explicit IndexedInstallOperationsDstComparator(Graph* graph)
87     : graph_(graph) {}
88 
89   // Compares the operations in the vertex a and b of graph_.
operator ()(size_t a,size_t b) const90   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 
CheckGraph(const Graph & graph)101 void InplaceGenerator::CheckGraph(const Graph& graph) {
102   for (const Vertex& v : graph) {
103     CHECK(v.aop.op.has_type());
104   }
105 }
106 
SubstituteBlocks(Vertex * vertex,const vector<Extent> & remove_extents,const vector<Extent> & replace_extents)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 
CutEdges(Graph * graph,const set<Edge> & edges,vector<CutEdgeVertexes> * out_cuts)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.
CreateEdges(Graph * graph,const vector<Block> & blocks)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:
SortCutsByTopoOrderLess(const vector<vector<Vertex::Index>::size_type> & table)237   explicit SortCutsByTopoOrderLess(
238       const vector<vector<Vertex::Index>::size_type>& table)
239       : table_(table) {}
operator ()(const CutEdgeVertexes & a,const CutEdgeVertexes & b)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 
GenerateReverseTopoOrderMap(const vector<Vertex::Index> & op_indexes,vector<vector<Vertex::Index>::size_type> * reverse_op_indexes)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 
SortCutsByTopoOrder(const vector<Vertex::Index> & op_indexes,vector<CutEdgeVertexes> * cuts)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 
MoveAndSortFullOpsToBack(Graph * graph,vector<Vertex::Index> * op_indexes)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>
TempBlocksExistInExtents(const T & extents)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.
ConvertCutsToFull(Graph * graph,const string & new_part,BlobFileWriter * blob_file,vector<Vertex::Index> * op_indexes,vector<vector<Vertex::Index>::size_type> * reverse_op_indexes,const vector<CutEdgeVertexes> & cuts)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.
AssignBlockForAdjoiningCuts(Graph * graph,const string & new_part,BlobFileWriter * blob_file,vector<Vertex::Index> * op_indexes,vector<vector<Vertex::Index>::size_type> * reverse_op_indexes,const vector<CutEdgeVertexes> & cuts)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 
AssignTempBlocks(Graph * graph,const string & new_part,BlobFileWriter * blob_file,vector<Vertex::Index> * op_indexes,vector<vector<Vertex::Index>::size_type> * reverse_op_indexes,const vector<CutEdgeVertexes> & cuts)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 
NoTempBlocksRemain(const Graph & graph)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 
ConvertCutToFullOp(Graph * graph,const CutEdgeVertexes & cut,const string & new_part,BlobFileWriter * blob_file)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 
ConvertGraphToDag(Graph * graph,const string & new_part,BlobFileWriter * blob_file,vector<Vertex::Index> * final_order,Vertex::Index scratch_vertex)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 
CreateScratchNode(uint64_t start_block,uint64_t num_blocks,Vertex * vertex)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 
AddInstallOpToBlocksVector(const InstallOperation & operation,const Graph & graph,Vertex::Index vertex,vector<Block> * blocks)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 
AddInstallOpToGraph(Graph * graph,Vertex::Index existing_vertex,vector<Block> * blocks,const InstallOperation & operation,const string & op_name)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 
ApplyMap(vector<uint64_t> * collection,const map<uint64_t,uint64_t> & the_map)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 
ResolveReadAfterWriteDependencies(const PartitionConfig & old_part,const PartitionConfig & new_part,uint64_t partition_size,size_t block_size,BlobFileWriter * blob_file,vector<AnnotatedOperation> * aops)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 
GenerateOperations(const PayloadGenerationConfig & config,const PartitionConfig & old_part,const PartitionConfig & new_part,BlobFileWriter * blob_file,vector<AnnotatedOperation> * aops)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