blob: 8ad4cde036545e67876fc0bf57244bc48919a601 [file] [log] [blame]
Alex Deymoa28e0192017-09-08 14:21:05 +02001// Copyright 2017 The Chromium OS Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5#ifndef _BSDIFF_PATCH_WRITER_H_
6#define _BSDIFF_PATCH_WRITER_H_
7
Tianjie Xu1c26e2e2017-10-26 17:19:41 -07008#include <memory>
Alex Deymoa28e0192017-09-08 14:21:05 +02009#include <string>
10#include <vector>
11
Tianjie Xu1c26e2e2017-10-26 17:19:41 -070012#include "bsdiff/compressor_interface.h"
Alex Deymo538a75d2017-09-27 15:34:59 +020013#include "bsdiff/patch_writer_interface.h"
Alex Deymoa28e0192017-09-08 14:21:05 +020014
15namespace bsdiff {
16
Tianjie Xu1f1cdb22017-11-20 11:05:55 -080017// A PatchWriterInterface class with three compressors and a 32-byte header.
Alex Deymo538a75d2017-09-27 15:34:59 +020018class BsdiffPatchWriter : public PatchWriterInterface {
Alex Deymoa28e0192017-09-08 14:21:05 +020019 public:
Tianjie Xu1f1cdb22017-11-20 11:05:55 -080020 // Create the patch writer using the upstream's "BSDIFF40" format. It uses
21 // bz2 as the compression algorithm and the file |patch_filename| to write
22 // the patch data.
23 explicit BsdiffPatchWriter(const std::string& patch_filename);
24
25 // Create the patch writer using the "BSDF2" format. It uses the compressor
Tianjie Xu2e70b552018-03-02 16:22:10 -080026 // with algorithm |type|; and quality |brotli_quality| if it's brotli. This
27 // writer also writes the patch data to the file |patch_filename|.
Tianjie Xu1f1cdb22017-11-20 11:05:55 -080028 BsdiffPatchWriter(const std::string& patch_filename,
Tianjie Xu77833b62018-03-07 18:13:47 -080029 const std::vector<CompressorType>& types,
Tianjie Xu2e70b552018-03-02 16:22:10 -080030 int brotli_quality);
Alex Deymoa28e0192017-09-08 14:21:05 +020031
Alex Deymo538a75d2017-09-27 15:34:59 +020032 // PatchWriterInterface overrides.
Alex Deymo4dadd8b2017-10-26 16:19:33 +020033 bool Init(size_t new_size) override;
Alex Deymo68c0e7f2017-10-02 20:38:12 +020034 bool WriteDiffStream(const uint8_t* data, size_t size) override;
35 bool WriteExtraStream(const uint8_t* data, size_t size) override;
Alex Deymo538a75d2017-09-27 15:34:59 +020036 bool AddControlEntry(const ControlEntry& entry) override;
37 bool Close() override;
Alex Deymoa28e0192017-09-08 14:21:05 +020038
39 private:
Tianjie Xu32b1f212018-03-06 11:42:45 -080040 // Add supported compressors to |compressor_list|; return false if we failed
41 // to initialize one of them.
42 bool InitializeCompressorList(
43 std::vector<std::unique_ptr<CompressorInterface>>* compressor_list);
44
45 // Select the compressor in |compressor_list| that produces the smallest
46 // patch, and put the result in |smallest_compressor|.
47 bool SelectSmallestResult(
48 const std::vector<std::unique_ptr<CompressorInterface>>& compressor_list,
49 CompressorInterface** smallest_compressor);
50
51
52 // Write the BSDIFF patch header to the |fp_|.
53 // Arguments:
54 // A three bytes array with the compressor types of ctrl|diff|extra stream
55 // Size of the compressed control block
56 // Size of the compressed diff block.
57 bool WriteHeader(uint8_t types[3], uint64_t ctrl_size, uint64_t diff_size);
Alex Deymoa28e0192017-09-08 14:21:05 +020058
Alex Deymo68c0e7f2017-10-02 20:38:12 +020059 // Bytes of the new files already written. Needed to store the new length in
60 // the header of the file.
Alex Deymoa28e0192017-09-08 14:21:05 +020061 uint64_t written_output_{0};
62
Alex Deymoa28e0192017-09-08 14:21:05 +020063 // The current file we are writing to.
64 FILE* fp_{nullptr};
Alex Deymo538a75d2017-09-27 15:34:59 +020065 std::string patch_filename_;
Alex Deymoa28e0192017-09-08 14:21:05 +020066
Tianjie Xub4cba642017-11-14 22:46:38 -080067 // The format of bsdiff we're using.
68 BsdiffFormat format_;
69
Tianjie Xu77833b62018-03-07 18:13:47 -080070 // The compressors we're using.
71 std::vector<CompressorType> types_;
Tianjie Xu32b1f212018-03-06 11:42:45 -080072
73 // The compression quality of the brotli compressor.
74 int brotli_quality_;
75
76 // The list of compressors to try for each stream.
77 std::vector<std::unique_ptr<CompressorInterface>> ctrl_stream_list_;
78 std::vector<std::unique_ptr<CompressorInterface>> diff_stream_list_;
79 std::vector<std::unique_ptr<CompressorInterface>> extra_stream_list_;
Alex Deymoa28e0192017-09-08 14:21:05 +020080};
81
82} // namespace bsdiff
83
84#endif // _BSDIFF_PATCH_WRITER_H_