blob: 1aaafd5b10062403ce4f14fcb645f29846cf2a1a [file] [log] [blame]
//
// 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 <memory>
#include <string>
#include <verity/hash_tree_builder.h>
#include <base/logging.h>
#include <base/posix/eintr_wrapper.h>
#include <fec/ecc.h>
extern "C" {
#include <fec.h>
}
#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_;
size_t current_round_;
size_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_;
size_t rs_n_;
bool verify_mode_;
std::unique_ptr<void, decltype(&free_rs_char)> 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<HashTreeBuilder> 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_