// // Copyright (C) 2018 The Android Open Source Project // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. // #ifndef UPDATE_ENGINE_PAYLOAD_CONSUMER_VERITY_WRITER_ANDROID_H_ #define UPDATE_ENGINE_PAYLOAD_CONSUMER_VERITY_WRITER_ANDROID_H_ #include #include #include #include #include #include extern "C" { #include } #include "payload_consumer/file_descriptor.h" #include "update_engine/payload_consumer/cached_file_descriptor.h" #include "update_engine/payload_consumer/verity_writer_interface.h" namespace chromeos_update_engine { enum class EncodeFECStep { kInitFDStep, kEncodeRoundStep, kWriteStep, kComplete }; class IncrementalEncodeFEC { public: IncrementalEncodeFEC() : rs_char_(nullptr, &free_rs_char), cache_fd_(nullptr, 1 * (1 << 20)) {} // Initialize all member variables needed to performe FEC Computation bool Init(const uint64_t _data_offset, const uint64_t _data_size, const uint64_t _fec_offset, const uint64_t _fec_size, const uint64_t _fec_roots, const uint64_t _block_size, const bool _verify_mode); bool Compute(FileDescriptor* _read_fd, FileDescriptor* _write_fd); void UpdateState(); bool Finished() const; void Reset(); double ReportProgress() const; private: brillo::Blob rs_blocks_; brillo::Blob buffer_; brillo::Blob fec_; brillo::Blob fec_read_; EncodeFECStep current_step_; uint64_t current_round_; uint64_t num_rounds_; FileDescriptor* read_fd_; FileDescriptor* write_fd_; uint64_t data_offset_; uint64_t data_size_; uint64_t fec_offset_; uint64_t fec_size_; uint64_t fec_roots_; uint64_t block_size_; uint64_t rs_n_; bool verify_mode_; std::unique_ptr rs_char_; UnownedCachedFileDescriptor cache_fd_; }; class VerityWriterAndroid : public VerityWriterInterface { public: VerityWriterAndroid() = default; ~VerityWriterAndroid() override = default; bool Init(const InstallPlan::Partition& partition); bool Update(uint64_t offset, const uint8_t* buffer, size_t size) override; bool Finalize(FileDescriptor* read_fd, FileDescriptor* write_fd) override; bool IncrementalFinalize(FileDescriptor* read_fd, FileDescriptor* write_fd) override; double GetProgress() override; bool FECFinished() const override; // Read [data_offset : data_offset + data_size) from |path| and encode FEC // data, if |verify_mode|, then compare the encoded FEC with the one in // |path|, otherwise write the encoded FEC to |path|. We can't encode as we go // in each Update() like hash tree, because for every rs block, its data are // spreaded across entire |data_size|, unless we can cache all data in // memory, we have to re-read them from disk. static bool EncodeFEC(FileDescriptor* read_fd, FileDescriptor* write_fd, uint64_t data_offset, uint64_t data_size, uint64_t fec_offset, uint64_t fec_size, uint32_t fec_roots, uint32_t block_size, bool verify_mode); static bool EncodeFEC(const std::string& path, uint64_t data_offset, uint64_t data_size, uint64_t fec_offset, uint64_t fec_size, uint32_t fec_roots, uint32_t block_size, bool verify_mode); private: // stores the state of EncodeFEC IncrementalEncodeFEC encodeFEC_; bool hash_tree_written_ = false; const InstallPlan::Partition* partition_ = nullptr; std::unique_ptr hash_tree_builder_; uint64_t total_offset_ = 0; DISALLOW_COPY_AND_ASSIGN(VerityWriterAndroid); }; } // namespace chromeos_update_engine #endif // UPDATE_ENGINE_PAYLOAD_CONSUMER_VERITY_WRITER_ANDROID_H_