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