xref: /aosp_15_r20/system/update_engine/lz4diff/lz4diff.cc (revision 5a9231315b4521097b8dc3750bc806fcafe0c72f)
1*5a923131SAndroid Build Coastguard Worker //
2*5a923131SAndroid Build Coastguard Worker // Copyright (C) 2021 The Android Open Source Project
3*5a923131SAndroid Build Coastguard Worker //
4*5a923131SAndroid Build Coastguard Worker // Licensed under the Apache License, Version 2.0 (the "License");
5*5a923131SAndroid Build Coastguard Worker // you may not use this file except in compliance with the License.
6*5a923131SAndroid Build Coastguard Worker // You may obtain a copy of the License at
7*5a923131SAndroid Build Coastguard Worker //
8*5a923131SAndroid Build Coastguard Worker //      http://www.apache.org/licenses/LICENSE-2.0
9*5a923131SAndroid Build Coastguard Worker //
10*5a923131SAndroid Build Coastguard Worker // Unless required by applicable law or agreed to in writing, software
11*5a923131SAndroid Build Coastguard Worker // distributed under the License is distributed on an "AS IS" BASIS,
12*5a923131SAndroid Build Coastguard Worker // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*5a923131SAndroid Build Coastguard Worker // See the License for the specific language governing permissions and
14*5a923131SAndroid Build Coastguard Worker // limitations under the License.
15*5a923131SAndroid Build Coastguard Worker //
16*5a923131SAndroid Build Coastguard Worker 
17*5a923131SAndroid Build Coastguard Worker #include "lz4diff.h"
18*5a923131SAndroid Build Coastguard Worker #include "lz4diff_compress.h"
19*5a923131SAndroid Build Coastguard Worker 
20*5a923131SAndroid Build Coastguard Worker #include <bsdiff/bsdiff.h>
21*5a923131SAndroid Build Coastguard Worker #include <bsdiff/constants.h>
22*5a923131SAndroid Build Coastguard Worker #include <bsdiff/patch_writer_factory.h>
23*5a923131SAndroid Build Coastguard Worker #include <bsdiff/patch_writer.h>
24*5a923131SAndroid Build Coastguard Worker #include <puffin/common.h>
25*5a923131SAndroid Build Coastguard Worker #include <puffin/puffdiff.h>
26*5a923131SAndroid Build Coastguard Worker #include <lz4.h>
27*5a923131SAndroid Build Coastguard Worker #include <lz4hc.h>
28*5a923131SAndroid Build Coastguard Worker 
29*5a923131SAndroid Build Coastguard Worker #include "update_engine/common/utils.h"
30*5a923131SAndroid Build Coastguard Worker #include "update_engine/common/hash_calculator.h"
31*5a923131SAndroid Build Coastguard Worker #include "update_engine/payload_generator/deflate_utils.h"
32*5a923131SAndroid Build Coastguard Worker #include "update_engine/payload_generator/delta_diff_generator.h"
33*5a923131SAndroid Build Coastguard Worker #include "lz4diff/lz4diff.pb.h"
34*5a923131SAndroid Build Coastguard Worker #include "lz4diff_format.h"
35*5a923131SAndroid Build Coastguard Worker 
36*5a923131SAndroid Build Coastguard Worker namespace chromeos_update_engine {
37*5a923131SAndroid Build Coastguard Worker 
StoreDstCompressedFileInfo(std::string_view recompressed_blob,std::string_view target_blob,const CompressedFile & dst_file_info,Lz4diffHeader * output)38*5a923131SAndroid Build Coastguard Worker bool StoreDstCompressedFileInfo(std::string_view recompressed_blob,
39*5a923131SAndroid Build Coastguard Worker                                 std::string_view target_blob,
40*5a923131SAndroid Build Coastguard Worker                                 const CompressedFile& dst_file_info,
41*5a923131SAndroid Build Coastguard Worker                                 Lz4diffHeader* output) {
42*5a923131SAndroid Build Coastguard Worker   *output->mutable_dst_info()->mutable_algo() = dst_file_info.algo;
43*5a923131SAndroid Build Coastguard Worker   output->mutable_dst_info()->set_zero_padding_enabled(
44*5a923131SAndroid Build Coastguard Worker       dst_file_info.zero_padding_enabled);
45*5a923131SAndroid Build Coastguard Worker   const auto& block_info = dst_file_info.blocks;
46*5a923131SAndroid Build Coastguard Worker   auto& dst_block_info = *output->mutable_dst_info()->mutable_block_info();
47*5a923131SAndroid Build Coastguard Worker   dst_block_info.Clear();
48*5a923131SAndroid Build Coastguard Worker   size_t offset = 0;
49*5a923131SAndroid Build Coastguard Worker   for (const auto& block : block_info) {
50*5a923131SAndroid Build Coastguard Worker     auto& pb_block = *dst_block_info.Add();
51*5a923131SAndroid Build Coastguard Worker     pb_block.set_uncompressed_offset(block.uncompressed_offset);
52*5a923131SAndroid Build Coastguard Worker     pb_block.set_uncompressed_length(block.uncompressed_length);
53*5a923131SAndroid Build Coastguard Worker     pb_block.set_compressed_length(block.compressed_length);
54*5a923131SAndroid Build Coastguard Worker     CHECK_LT(offset, recompressed_blob.size());
55*5a923131SAndroid Build Coastguard Worker     auto s1 = recompressed_blob.substr(offset, block.compressed_length);
56*5a923131SAndroid Build Coastguard Worker     auto s2 = target_blob.substr(offset, block.compressed_length);
57*5a923131SAndroid Build Coastguard Worker     if (s1 != s2) {
58*5a923131SAndroid Build Coastguard Worker       ScopedTempFile patch;
59*5a923131SAndroid Build Coastguard Worker       int err =
60*5a923131SAndroid Build Coastguard Worker           bsdiff::bsdiff(reinterpret_cast<const unsigned char*>(s1.data()),
61*5a923131SAndroid Build Coastguard Worker                          s1.size(),
62*5a923131SAndroid Build Coastguard Worker                          reinterpret_cast<const unsigned char*>(s2.data()),
63*5a923131SAndroid Build Coastguard Worker                          s2.size(),
64*5a923131SAndroid Build Coastguard Worker                          patch.path().c_str(),
65*5a923131SAndroid Build Coastguard Worker                          nullptr);
66*5a923131SAndroid Build Coastguard Worker       CHECK_EQ(err, 0);
67*5a923131SAndroid Build Coastguard Worker       LOG(WARNING) << "Recompress Postfix patch size: "
68*5a923131SAndroid Build Coastguard Worker                    << utils::FileSize(patch.path());
69*5a923131SAndroid Build Coastguard Worker       std::string patch_content;
70*5a923131SAndroid Build Coastguard Worker       TEST_AND_RETURN_FALSE(utils::ReadFile(patch.path(), &patch_content));
71*5a923131SAndroid Build Coastguard Worker       pb_block.set_postfix_bspatch(std::move(patch_content));
72*5a923131SAndroid Build Coastguard Worker     }
73*5a923131SAndroid Build Coastguard Worker     // Include recompressed blob hash, so we can determine if the device
74*5a923131SAndroid Build Coastguard Worker     // produces same compressed output
75*5a923131SAndroid Build Coastguard Worker     Blob recompressed_blob_hash;
76*5a923131SAndroid Build Coastguard Worker     TEST_AND_RETURN_FALSE(HashCalculator::RawHashOfBytes(
77*5a923131SAndroid Build Coastguard Worker         s1.data(), s1.length(), &recompressed_blob_hash));
78*5a923131SAndroid Build Coastguard Worker     pb_block.set_sha256_hash(recompressed_blob_hash.data(),
79*5a923131SAndroid Build Coastguard Worker                              recompressed_blob_hash.size());
80*5a923131SAndroid Build Coastguard Worker 
81*5a923131SAndroid Build Coastguard Worker     offset += block.compressed_length;
82*5a923131SAndroid Build Coastguard Worker   }
83*5a923131SAndroid Build Coastguard Worker   return true;
84*5a923131SAndroid Build Coastguard Worker }
85*5a923131SAndroid Build Coastguard Worker 
86*5a923131SAndroid Build Coastguard Worker template <typename Blob>
TryBsdiff(Blob src,Blob dst,Blob * output)87*5a923131SAndroid Build Coastguard Worker static bool TryBsdiff(Blob src, Blob dst, Blob* output) noexcept {
88*5a923131SAndroid Build Coastguard Worker   static constexpr auto kLz4diffDefaultBrotliQuality = 9;
89*5a923131SAndroid Build Coastguard Worker   CHECK_NE(output, nullptr);
90*5a923131SAndroid Build Coastguard Worker   ScopedTempFile patch;
91*5a923131SAndroid Build Coastguard Worker 
92*5a923131SAndroid Build Coastguard Worker   Blob bsdiff_delta;
93*5a923131SAndroid Build Coastguard Worker   bsdiff::BsdiffPatchWriter patch_writer(patch.path(),
94*5a923131SAndroid Build Coastguard Worker                                          {bsdiff::CompressorType::kBrotli},
95*5a923131SAndroid Build Coastguard Worker                                          kLz4diffDefaultBrotliQuality);
96*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(0 == bsdiff::bsdiff(src.data(),
97*5a923131SAndroid Build Coastguard Worker                                             src.size(),
98*5a923131SAndroid Build Coastguard Worker                                             dst.data(),
99*5a923131SAndroid Build Coastguard Worker                                             dst.size(),
100*5a923131SAndroid Build Coastguard Worker                                             &patch_writer,
101*5a923131SAndroid Build Coastguard Worker                                             nullptr));
102*5a923131SAndroid Build Coastguard Worker 
103*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(utils::ReadFile(patch.path(), &bsdiff_delta));
104*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(!bsdiff_delta.empty());
105*5a923131SAndroid Build Coastguard Worker   *output = std::move(bsdiff_delta);
106*5a923131SAndroid Build Coastguard Worker   return true;
107*5a923131SAndroid Build Coastguard Worker }
108*5a923131SAndroid Build Coastguard Worker 
TryFindDeflates(puffin::Buffer data,std::vector<puffin::BitExtent> * deflates)109*5a923131SAndroid Build Coastguard Worker bool TryFindDeflates(puffin::Buffer data,
110*5a923131SAndroid Build Coastguard Worker                      std::vector<puffin::BitExtent>* deflates) {
111*5a923131SAndroid Build Coastguard Worker   if (puffin::LocateDeflatesInZipArchive(data, deflates)) {
112*5a923131SAndroid Build Coastguard Worker     return true;
113*5a923131SAndroid Build Coastguard Worker   }
114*5a923131SAndroid Build Coastguard Worker   deflates->clear();
115*5a923131SAndroid Build Coastguard Worker   if (puffin::LocateDeflatesInGzip(data, deflates)) {
116*5a923131SAndroid Build Coastguard Worker     return true;
117*5a923131SAndroid Build Coastguard Worker   }
118*5a923131SAndroid Build Coastguard Worker   deflates->clear();
119*5a923131SAndroid Build Coastguard Worker   return false;
120*5a923131SAndroid Build Coastguard Worker }
121*5a923131SAndroid Build Coastguard Worker 
ConstructLz4diffPatch(Blob inner_patch,const Lz4diffHeader & header,Blob * output)122*5a923131SAndroid Build Coastguard Worker static bool ConstructLz4diffPatch(Blob inner_patch,
123*5a923131SAndroid Build Coastguard Worker                                   const Lz4diffHeader& header,
124*5a923131SAndroid Build Coastguard Worker                                   Blob* output) {
125*5a923131SAndroid Build Coastguard Worker   Blob patch(kLz4diffHeaderSize);
126*5a923131SAndroid Build Coastguard Worker   std::memcpy(patch.data(), kLz4diffMagic.data(), kLz4diffMagic.size());
127*5a923131SAndroid Build Coastguard Worker   *reinterpret_cast<uint32_t*>(patch.data() + kLz4diffMagic.size()) =
128*5a923131SAndroid Build Coastguard Worker       htobe32(kLz4diffVersion);
129*5a923131SAndroid Build Coastguard Worker 
130*5a923131SAndroid Build Coastguard Worker   std::string serialized_pb;
131*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(header.SerializeToString(&serialized_pb));
132*5a923131SAndroid Build Coastguard Worker   *reinterpret_cast<uint32_t*>(patch.data() + kLz4diffMagic.size() + 4) =
133*5a923131SAndroid Build Coastguard Worker       htobe32(serialized_pb.size());
134*5a923131SAndroid Build Coastguard Worker   patch.insert(patch.end(), serialized_pb.begin(), serialized_pb.end());
135*5a923131SAndroid Build Coastguard Worker   patch.insert(patch.end(), inner_patch.begin(), inner_patch.end());
136*5a923131SAndroid Build Coastguard Worker 
137*5a923131SAndroid Build Coastguard Worker   *output = std::move(patch);
138*5a923131SAndroid Build Coastguard Worker   return true;
139*5a923131SAndroid Build Coastguard Worker }
140*5a923131SAndroid Build Coastguard Worker 
TryPuffdiff(puffin::Buffer src,puffin::Buffer dst,Blob * output)141*5a923131SAndroid Build Coastguard Worker static bool TryPuffdiff(puffin::Buffer src,
142*5a923131SAndroid Build Coastguard Worker                         puffin::Buffer dst,
143*5a923131SAndroid Build Coastguard Worker                         Blob* output) noexcept {
144*5a923131SAndroid Build Coastguard Worker   CHECK_NE(output, nullptr);
145*5a923131SAndroid Build Coastguard Worker   std::vector<puffin::BitExtent> src_deflates;
146*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(TryFindDeflates(src, &src_deflates));
147*5a923131SAndroid Build Coastguard Worker   std::vector<puffin::BitExtent> dst_deflates;
148*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(TryFindDeflates(dst, &dst_deflates));
149*5a923131SAndroid Build Coastguard Worker   if (src_deflates.empty() || dst_deflates.empty()) {
150*5a923131SAndroid Build Coastguard Worker     return false;
151*5a923131SAndroid Build Coastguard Worker   }
152*5a923131SAndroid Build Coastguard Worker 
153*5a923131SAndroid Build Coastguard Worker   Blob puffdiff_delta;
154*5a923131SAndroid Build Coastguard Worker   ScopedTempFile temp_file("puffdiff-delta.XXXXXX");
155*5a923131SAndroid Build Coastguard Worker   // Perform PuffDiff operation.
156*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(puffin::PuffDiff(
157*5a923131SAndroid Build Coastguard Worker       src, dst, src_deflates, dst_deflates, temp_file.path(), &puffdiff_delta));
158*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(!puffdiff_delta.empty());
159*5a923131SAndroid Build Coastguard Worker 
160*5a923131SAndroid Build Coastguard Worker   *output = std::move(puffdiff_delta);
161*5a923131SAndroid Build Coastguard Worker   return true;
162*5a923131SAndroid Build Coastguard Worker }
163*5a923131SAndroid Build Coastguard Worker 
StoreSrcCompressedFileInfo(const CompressedFile & src_file_info,Lz4diffHeader * header)164*5a923131SAndroid Build Coastguard Worker static void StoreSrcCompressedFileInfo(const CompressedFile& src_file_info,
165*5a923131SAndroid Build Coastguard Worker                                        Lz4diffHeader* header) {
166*5a923131SAndroid Build Coastguard Worker   *header->mutable_src_info()->mutable_algo() = src_file_info.algo;
167*5a923131SAndroid Build Coastguard Worker   header->mutable_src_info()->set_zero_padding_enabled(
168*5a923131SAndroid Build Coastguard Worker       src_file_info.zero_padding_enabled);
169*5a923131SAndroid Build Coastguard Worker   auto& src_blocks = *header->mutable_src_info()->mutable_block_info();
170*5a923131SAndroid Build Coastguard Worker   src_blocks.Clear();
171*5a923131SAndroid Build Coastguard Worker   for (const auto& block : src_file_info.blocks) {
172*5a923131SAndroid Build Coastguard Worker     auto& block_info = *src_blocks.Add();
173*5a923131SAndroid Build Coastguard Worker     block_info.set_uncompressed_length(block.uncompressed_length);
174*5a923131SAndroid Build Coastguard Worker     block_info.set_uncompressed_offset(block.uncompressed_offset);
175*5a923131SAndroid Build Coastguard Worker     block_info.set_compressed_length(block.compressed_length);
176*5a923131SAndroid Build Coastguard Worker   }
177*5a923131SAndroid Build Coastguard Worker   return;
178*5a923131SAndroid Build Coastguard Worker }
179*5a923131SAndroid Build Coastguard Worker 
Lz4Diff(std::string_view src,std::string_view dst,const CompressedFile & src_file_info,const CompressedFile & dst_file_info,Blob * output,InstallOperation::Type * op_type)180*5a923131SAndroid Build Coastguard Worker bool Lz4Diff(std::string_view src,
181*5a923131SAndroid Build Coastguard Worker              std::string_view dst,
182*5a923131SAndroid Build Coastguard Worker              const CompressedFile& src_file_info,
183*5a923131SAndroid Build Coastguard Worker              const CompressedFile& dst_file_info,
184*5a923131SAndroid Build Coastguard Worker              Blob* output,
185*5a923131SAndroid Build Coastguard Worker              InstallOperation::Type* op_type) noexcept {
186*5a923131SAndroid Build Coastguard Worker   const auto& src_block_info = src_file_info.blocks;
187*5a923131SAndroid Build Coastguard Worker   const auto& dst_block_info = dst_file_info.blocks;
188*5a923131SAndroid Build Coastguard Worker 
189*5a923131SAndroid Build Coastguard Worker   auto decompressed_src = TryDecompressBlob(
190*5a923131SAndroid Build Coastguard Worker       src, src_block_info, src_file_info.zero_padding_enabled);
191*5a923131SAndroid Build Coastguard Worker   auto decompressed_dst = TryDecompressBlob(
192*5a923131SAndroid Build Coastguard Worker       dst, dst_block_info, dst_file_info.zero_padding_enabled);
193*5a923131SAndroid Build Coastguard Worker   if (decompressed_src.empty() || decompressed_dst.empty()) {
194*5a923131SAndroid Build Coastguard Worker     LOG(ERROR) << "Failed to decompress input data";
195*5a923131SAndroid Build Coastguard Worker     return false;
196*5a923131SAndroid Build Coastguard Worker   }
197*5a923131SAndroid Build Coastguard Worker 
198*5a923131SAndroid Build Coastguard Worker   Lz4diffHeader header;
199*5a923131SAndroid Build Coastguard Worker   // BSDIFF isn't supposed to fail, so return error if BSDIFF failed.
200*5a923131SAndroid Build Coastguard Worker   Blob patch_data;
201*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(
202*5a923131SAndroid Build Coastguard Worker       TryBsdiff(decompressed_src, decompressed_dst, &patch_data));
203*5a923131SAndroid Build Coastguard Worker   header.set_inner_type(InnerPatchType::BSDIFF);
204*5a923131SAndroid Build Coastguard Worker   if (op_type) {
205*5a923131SAndroid Build Coastguard Worker     *op_type = InstallOperation::LZ4DIFF_BSDIFF;
206*5a923131SAndroid Build Coastguard Worker   }
207*5a923131SAndroid Build Coastguard Worker   // PUFFDIFF might fail, as the input data might not be deflate compressed.
208*5a923131SAndroid Build Coastguard Worker 
209*5a923131SAndroid Build Coastguard Worker   Blob puffdiff_delta;
210*5a923131SAndroid Build Coastguard Worker   if (TryPuffdiff(decompressed_src, decompressed_dst, &puffdiff_delta) &&
211*5a923131SAndroid Build Coastguard Worker       puffdiff_delta.size() < patch_data.size()) {
212*5a923131SAndroid Build Coastguard Worker     patch_data = std::move(puffdiff_delta);
213*5a923131SAndroid Build Coastguard Worker     header.set_inner_type(InnerPatchType::PUFFDIFF);
214*5a923131SAndroid Build Coastguard Worker     if (op_type) {
215*5a923131SAndroid Build Coastguard Worker       *op_type = InstallOperation::LZ4DIFF_PUFFDIFF;
216*5a923131SAndroid Build Coastguard Worker     }
217*5a923131SAndroid Build Coastguard Worker   }
218*5a923131SAndroid Build Coastguard Worker   // Free up memory used by |decompressed_src| , as we don't need it anymore.
219*5a923131SAndroid Build Coastguard Worker   decompressed_src = {};
220*5a923131SAndroid Build Coastguard Worker 
221*5a923131SAndroid Build Coastguard Worker   auto recompressed_blob = TryCompressBlob(ToStringView(decompressed_dst),
222*5a923131SAndroid Build Coastguard Worker                                            dst_block_info,
223*5a923131SAndroid Build Coastguard Worker                                            dst_file_info.zero_padding_enabled,
224*5a923131SAndroid Build Coastguard Worker                                            dst_file_info.algo);
225*5a923131SAndroid Build Coastguard Worker   TEST_AND_RETURN_FALSE(recompressed_blob.size() > 0);
226*5a923131SAndroid Build Coastguard Worker 
227*5a923131SAndroid Build Coastguard Worker   StoreSrcCompressedFileInfo(src_file_info, &header);
228*5a923131SAndroid Build Coastguard Worker   StoreDstCompressedFileInfo(
229*5a923131SAndroid Build Coastguard Worker       ToStringView(recompressed_blob), dst, dst_file_info, &header);
230*5a923131SAndroid Build Coastguard Worker   return ConstructLz4diffPatch(std::move(patch_data), header, output);
231*5a923131SAndroid Build Coastguard Worker }
232*5a923131SAndroid Build Coastguard Worker 
Lz4Diff(const Blob & src,const Blob & dst,const CompressedFile & src_file_info,const CompressedFile & dst_file_info,Blob * output,InstallOperation::Type * op_type)233*5a923131SAndroid Build Coastguard Worker bool Lz4Diff(const Blob& src,
234*5a923131SAndroid Build Coastguard Worker              const Blob& dst,
235*5a923131SAndroid Build Coastguard Worker              const CompressedFile& src_file_info,
236*5a923131SAndroid Build Coastguard Worker              const CompressedFile& dst_file_info,
237*5a923131SAndroid Build Coastguard Worker              Blob* output,
238*5a923131SAndroid Build Coastguard Worker              InstallOperation::Type* op_type) noexcept {
239*5a923131SAndroid Build Coastguard Worker   return Lz4Diff(ToStringView(src),
240*5a923131SAndroid Build Coastguard Worker                  ToStringView(dst),
241*5a923131SAndroid Build Coastguard Worker                  src_file_info,
242*5a923131SAndroid Build Coastguard Worker                  dst_file_info,
243*5a923131SAndroid Build Coastguard Worker                  output,
244*5a923131SAndroid Build Coastguard Worker                  op_type);
245*5a923131SAndroid Build Coastguard Worker }
246*5a923131SAndroid Build Coastguard Worker 
247*5a923131SAndroid Build Coastguard Worker }  // namespace chromeos_update_engine
248