blob: 83f40e07a057e7fc65cf008f2eeebcfc880f30f1 [file] [log] [blame]
Alex Deymoaea4c1c2015-08-19 20:24:43 -07001//
2// Copyright (C) 2009 The Android Open Source Project
3//
4// Licensed under the Apache License, Version 2.0 (the "License");
5// you may not use this file except in compliance with the License.
6// You may obtain a copy of the License at
7//
8// http://www.apache.org/licenses/LICENSE-2.0
9//
10// Unless required by applicable law or agreed to in writing, software
11// distributed under the License is distributed on an "AS IS" BASIS,
12// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13// See the License for the specific language governing permissions and
14// limitations under the License.
15//
Andrew de los Reyes80061062010-02-04 14:25:00 -080016
Gilad Arnoldcf175a02014-07-10 16:48:47 -070017#ifndef UPDATE_ENGINE_EXTENT_WRITER_H_
18#define UPDATE_ENGINE_EXTENT_WRITER_H_
Andrew de los Reyes80061062010-02-04 14:25:00 -080019
20#include <vector>
Alex Deymo8427b4a2014-11-05 14:00:32 -080021
22#include <base/logging.h>
Alex Vakulenko3f39d5c2015-10-13 09:27:13 -070023#include <brillo/secure_blob.h>
Alex Deymo8427b4a2014-11-05 14:00:32 -080024
Nam T. Nguyenf1d582e2014-12-08 15:07:17 -080025#include "update_engine/file_descriptor.h"
Andrew de los Reyes80061062010-02-04 14:25:00 -080026#include "update_engine/update_metadata.pb.h"
27#include "update_engine/utils.h"
28
29// ExtentWriter is an abstract class which synchronously writes to a given
30// file descriptor at the extents given.
31
32namespace chromeos_update_engine {
33
Andrew de los Reyes80061062010-02-04 14:25:00 -080034class ExtentWriter {
35 public:
Alex Deymo05322872015-09-30 09:50:24 -070036 ExtentWriter() = default;
Andrew de los Reyes80061062010-02-04 14:25:00 -080037 virtual ~ExtentWriter() {
38 LOG_IF(ERROR, !end_called_) << "End() not called on ExtentWriter.";
39 }
40
41 // Returns true on success.
Nam T. Nguyenf1d582e2014-12-08 15:07:17 -080042 virtual bool Init(FileDescriptorPtr fd,
Andrew de los Reyes80061062010-02-04 14:25:00 -080043 const std::vector<Extent>& extents,
Andrew de los Reyes08c4e272010-04-15 14:02:17 -070044 uint32_t block_size) = 0;
Andrew de los Reyes80061062010-02-04 14:25:00 -080045
46 // Returns true on success.
47 virtual bool Write(const void* bytes, size_t count) = 0;
48
49 // Should be called when all writing is complete. Returns true on success.
50 // The fd is not closed. Caller is responsible for closing it.
51 bool End() {
52 end_called_ = true;
53 return EndImpl();
54 }
55 virtual bool EndImpl() = 0;
56 private:
Alex Deymo05322872015-09-30 09:50:24 -070057 bool end_called_{false};
Andrew de los Reyes80061062010-02-04 14:25:00 -080058};
59
60// DirectExtentWriter is probably the simplest ExtentWriter implementation.
61// It writes the data directly into the extents.
62
63class DirectExtentWriter : public ExtentWriter {
64 public:
Alex Deymo05322872015-09-30 09:50:24 -070065 DirectExtentWriter() = default;
66 ~DirectExtentWriter() override = default;
Andrew de los Reyes80061062010-02-04 14:25:00 -080067
Nam T. Nguyenf1d582e2014-12-08 15:07:17 -080068 bool Init(FileDescriptorPtr fd,
69 const std::vector<Extent>& extents,
Alex Deymo05322872015-09-30 09:50:24 -070070 uint32_t block_size) override {
Andrew de los Reyes80061062010-02-04 14:25:00 -080071 fd_ = fd;
72 block_size_ = block_size;
73 extents_ = extents;
74 return true;
75 }
Alex Deymo05322872015-09-30 09:50:24 -070076 bool Write(const void* bytes, size_t count) override;
77 bool EndImpl() override { return true; }
Andrew de los Reyes80061062010-02-04 14:25:00 -080078
79 private:
Alex Deymo05322872015-09-30 09:50:24 -070080 FileDescriptorPtr fd_{nullptr};
Chris Masone4dc2ada2010-09-23 12:43:03 -070081
Alex Deymo05322872015-09-30 09:50:24 -070082 size_t block_size_{0};
Andrew de los Reyes80061062010-02-04 14:25:00 -080083 // Bytes written into next_extent_index_ thus far
Alex Deymo05322872015-09-30 09:50:24 -070084 uint64_t extent_bytes_written_{0};
Andrew de los Reyes80061062010-02-04 14:25:00 -080085 std::vector<Extent> extents_;
86 // The next call to write should correspond to extents_[next_extent_index_]
Alex Deymo05322872015-09-30 09:50:24 -070087 std::vector<Extent>::size_type next_extent_index_{0};
Andrew de los Reyes80061062010-02-04 14:25:00 -080088};
89
90// Takes an underlying ExtentWriter to which all operations are delegated.
91// When End() is called, ZeroPadExtentWriter ensures that the total number
92// of bytes written is a multiple of block_size_. If not, it writes zeros
93// to pad as needed.
94
95class ZeroPadExtentWriter : public ExtentWriter {
96 public:
Alex Deymo05322872015-09-30 09:50:24 -070097 explicit ZeroPadExtentWriter(
98 std::unique_ptr<ExtentWriter> underlying_extent_writer)
99 : underlying_extent_writer_(std::move(underlying_extent_writer)) {}
100 ~ZeroPadExtentWriter() override = default;
Andrew de los Reyes80061062010-02-04 14:25:00 -0800101
Nam T. Nguyenf1d582e2014-12-08 15:07:17 -0800102 bool Init(FileDescriptorPtr fd,
103 const std::vector<Extent>& extents,
Alex Deymo05322872015-09-30 09:50:24 -0700104 uint32_t block_size) override {
Andrew de los Reyes80061062010-02-04 14:25:00 -0800105 block_size_ = block_size;
106 return underlying_extent_writer_->Init(fd, extents, block_size);
107 }
Alex Deymo05322872015-09-30 09:50:24 -0700108 bool Write(const void* bytes, size_t count) override {
Andrew de los Reyes80061062010-02-04 14:25:00 -0800109 if (underlying_extent_writer_->Write(bytes, count)) {
110 bytes_written_mod_block_size_ += count;
111 bytes_written_mod_block_size_ %= block_size_;
112 return true;
113 }
114 return false;
115 }
Alex Deymo05322872015-09-30 09:50:24 -0700116 bool EndImpl() override {
Andrew de los Reyes80061062010-02-04 14:25:00 -0800117 if (bytes_written_mod_block_size_) {
118 const size_t write_size = block_size_ - bytes_written_mod_block_size_;
Alex Vakulenko3f39d5c2015-10-13 09:27:13 -0700119 brillo::Blob zeros(write_size, 0);
Alex Vakulenkof68bbbc2015-02-09 12:53:18 -0800120 TEST_AND_RETURN_FALSE(underlying_extent_writer_->Write(zeros.data(),
Andrew de los Reyes80061062010-02-04 14:25:00 -0800121 write_size));
122 }
123 return underlying_extent_writer_->End();
124 }
125
126 private:
Alex Deymo05322872015-09-30 09:50:24 -0700127 std::unique_ptr<ExtentWriter> underlying_extent_writer_;
128 size_t block_size_{0};
129 size_t bytes_written_mod_block_size_{0};
Andrew de los Reyes80061062010-02-04 14:25:00 -0800130};
131
132} // namespace chromeos_update_engine
133
Gilad Arnoldcf175a02014-07-10 16:48:47 -0700134#endif // UPDATE_ENGINE_EXTENT_WRITER_H_