Alex Deymo | aea4c1c | 2015-08-19 20:24:43 -0700 | [diff] [blame] | 1 | // |
| 2 | // Copyright (C) 2011 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 Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 16 | |
| 17 | #include <string.h> |
| 18 | #include <unistd.h> |
Darin Petkov | 7ed561b | 2011-10-04 02:59:03 -0700 | [diff] [blame] | 19 | |
Ben Chan | ab5a0af | 2017-10-12 14:57:50 -0700 | [diff] [blame] | 20 | #include <memory> |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 21 | #include <string> |
| 22 | #include <vector> |
Darin Petkov | 7ed561b | 2011-10-04 02:59:03 -0700 | [diff] [blame] | 23 | |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 24 | #include <gtest/gtest.h> |
Darin Petkov | 7ed561b | 2011-10-04 02:59:03 -0700 | [diff] [blame] | 25 | |
Alex Deymo | 39910dc | 2015-11-09 17:04:30 -0800 | [diff] [blame] | 26 | #include "update_engine/common/test_utils.h" |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 27 | #include "update_engine/payload_consumer/bzip_extent_writer.h" |
| 28 | #include "update_engine/payload_consumer/extent_writer.h" |
| 29 | #include "update_engine/payload_consumer/xz_extent_writer.h" |
Alex Deymo | 0bc2611 | 2015-10-19 20:54:57 -0700 | [diff] [blame] | 30 | #include "update_engine/payload_generator/bzip.h" |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 31 | #include "update_engine/payload_generator/xz.h" |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 32 | |
Alex Deymo | 10875d9 | 2014-11-10 21:52:57 -0800 | [diff] [blame] | 33 | using chromeos_update_engine::test_utils::kRandomString; |
Amin Hassani | cd7edbe | 2017-09-18 17:05:02 -0700 | [diff] [blame] | 34 | using google::protobuf::RepeatedPtrField; |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 35 | using std::string; |
| 36 | using std::vector; |
| 37 | |
| 38 | namespace chromeos_update_engine { |
| 39 | |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 40 | namespace { |
| 41 | |
| 42 | // ExtentWriter class that writes to memory, used to test the decompression |
| 43 | // step with the corresponding extent writer. |
| 44 | class MemoryExtentWriter : public ExtentWriter { |
| 45 | public: |
| 46 | // Creates the ExtentWriter that will write all the bytes to the passed |data| |
| 47 | // blob. |
| 48 | explicit MemoryExtentWriter(brillo::Blob* data) : data_(data) { |
| 49 | data_->clear(); |
| 50 | } |
| 51 | ~MemoryExtentWriter() override = default; |
| 52 | |
| 53 | bool Init(FileDescriptorPtr fd, |
Amin Hassani | cd7edbe | 2017-09-18 17:05:02 -0700 | [diff] [blame] | 54 | const RepeatedPtrField<Extent>& extents, |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 55 | uint32_t block_size) override { |
| 56 | return true; |
| 57 | } |
| 58 | bool Write(const void* bytes, size_t count) override { |
| 59 | data_->reserve(data_->size() + count); |
| 60 | data_->insert(data_->end(), |
| 61 | static_cast<const uint8_t*>(bytes), |
| 62 | static_cast<const uint8_t*>(bytes) + count); |
| 63 | return true; |
| 64 | } |
| 65 | bool EndImpl() override { return true; } |
| 66 | |
| 67 | private: |
| 68 | brillo::Blob* data_; |
| 69 | }; |
| 70 | |
| 71 | template <typename W> |
| 72 | bool DecompressWithWriter(const brillo::Blob& in, brillo::Blob* out) { |
| 73 | std::unique_ptr<ExtentWriter> writer( |
Ben Chan | ab5a0af | 2017-10-12 14:57:50 -0700 | [diff] [blame] | 74 | new W(std::make_unique<MemoryExtentWriter>(out))); |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 75 | // Init() parameters are ignored by the testing MemoryExtentWriter. |
Alex Deymo | d3251ef | 2016-03-29 21:01:54 -0700 | [diff] [blame] | 76 | bool ok = writer->Init(nullptr, {}, 1); |
| 77 | ok = writer->Write(in.data(), in.size()) && ok; |
| 78 | // Call End() even if the Write failed. |
| 79 | ok = writer->End() && ok; |
| 80 | return ok; |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 81 | } |
| 82 | |
| 83 | } // namespace |
| 84 | |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 85 | template <typename T> |
| 86 | class ZipTest : public ::testing::Test { |
| 87 | public: |
Alex Deymo | 0bc2611 | 2015-10-19 20:54:57 -0700 | [diff] [blame] | 88 | bool ZipCompress(const brillo::Blob& in, brillo::Blob* out) const = 0; |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 89 | bool ZipDecompress(const brillo::Blob& in, brillo::Blob* out) const = 0; |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 90 | }; |
| 91 | |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 92 | class BzipTest {}; |
| 93 | |
| 94 | template <> |
| 95 | class ZipTest<BzipTest> : public ::testing::Test { |
| 96 | public: |
Alex Deymo | 0bc2611 | 2015-10-19 20:54:57 -0700 | [diff] [blame] | 97 | bool ZipCompress(const brillo::Blob& in, brillo::Blob* out) const { |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 98 | return BzipCompress(in, out); |
| 99 | } |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 100 | bool ZipDecompress(const brillo::Blob& in, brillo::Blob* out) const { |
| 101 | return DecompressWithWriter<BzipExtentWriter>(in, out); |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 102 | } |
| 103 | }; |
| 104 | |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 105 | class XzTest {}; |
| 106 | |
| 107 | template <> |
| 108 | class ZipTest<XzTest> : public ::testing::Test { |
| 109 | public: |
| 110 | bool ZipCompress(const brillo::Blob& in, brillo::Blob* out) const { |
| 111 | return XzCompress(in, out); |
| 112 | } |
| 113 | bool ZipDecompress(const brillo::Blob& in, brillo::Blob* out) const { |
| 114 | return DecompressWithWriter<XzExtentWriter>(in, out); |
| 115 | } |
| 116 | }; |
| 117 | |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 118 | typedef ::testing::Types<BzipTest, XzTest> ZipTestTypes; |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 119 | |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 120 | TYPED_TEST_CASE(ZipTest, ZipTestTypes); |
| 121 | |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 122 | TYPED_TEST(ZipTest, SimpleTest) { |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 123 | string in_str( |
| 124 | "this should compress well xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx" |
| 125 | "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx" |
| 126 | "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx" |
| 127 | "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx" |
| 128 | "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx" |
| 129 | "xxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxxx"); |
| 130 | brillo::Blob in(in_str.begin(), in_str.end()); |
Alex Vakulenko | 3f39d5c | 2015-10-13 09:27:13 -0700 | [diff] [blame] | 131 | brillo::Blob out; |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 132 | EXPECT_TRUE(this->ZipCompress(in, &out)); |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 133 | EXPECT_LT(out.size(), in.size()); |
Alex Deymo | 80f70ff | 2016-02-10 16:08:11 -0800 | [diff] [blame] | 134 | EXPECT_GT(out.size(), 0U); |
Alex Vakulenko | 3f39d5c | 2015-10-13 09:27:13 -0700 | [diff] [blame] | 135 | brillo::Blob decompressed; |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 136 | EXPECT_TRUE(this->ZipDecompress(out, &decompressed)); |
| 137 | EXPECT_EQ(in.size(), decompressed.size()); |
Sen Jiang | 5288bd1 | 2018-05-03 11:39:04 -0700 | [diff] [blame^] | 138 | EXPECT_EQ(0, memcmp(in.data(), decompressed.data(), in.size())); |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 139 | } |
| 140 | |
| 141 | TYPED_TEST(ZipTest, PoorCompressionTest) { |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 142 | brillo::Blob in(std::begin(kRandomString), std::end(kRandomString)); |
Alex Vakulenko | 3f39d5c | 2015-10-13 09:27:13 -0700 | [diff] [blame] | 143 | brillo::Blob out; |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 144 | EXPECT_TRUE(this->ZipCompress(in, &out)); |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 145 | EXPECT_GT(out.size(), in.size()); |
Alex Vakulenko | 3f39d5c | 2015-10-13 09:27:13 -0700 | [diff] [blame] | 146 | brillo::Blob decompressed; |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 147 | EXPECT_TRUE(this->ZipDecompress(out, &decompressed)); |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 148 | EXPECT_EQ(in.size(), decompressed.size()); |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 149 | EXPECT_EQ(in, decompressed); |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 150 | } |
| 151 | |
| 152 | TYPED_TEST(ZipTest, MalformedZipTest) { |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 153 | brillo::Blob in(std::begin(kRandomString), std::end(kRandomString)); |
Alex Vakulenko | 3f39d5c | 2015-10-13 09:27:13 -0700 | [diff] [blame] | 154 | brillo::Blob out; |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 155 | EXPECT_FALSE(this->ZipDecompress(in, &out)); |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 156 | } |
| 157 | |
| 158 | TYPED_TEST(ZipTest, EmptyInputsTest) { |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 159 | brillo::Blob in; |
Alex Vakulenko | 3f39d5c | 2015-10-13 09:27:13 -0700 | [diff] [blame] | 160 | brillo::Blob out; |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 161 | EXPECT_TRUE(this->ZipDecompress(in, &out)); |
Alex Deymo | 80f70ff | 2016-02-10 16:08:11 -0800 | [diff] [blame] | 162 | EXPECT_EQ(0U, out.size()); |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 163 | |
Alex Deymo | 246bf21 | 2016-03-22 19:27:33 -0700 | [diff] [blame] | 164 | EXPECT_TRUE(this->ZipCompress(in, &out)); |
Alex Deymo | 80f70ff | 2016-02-10 16:08:11 -0800 | [diff] [blame] | 165 | EXPECT_EQ(0U, out.size()); |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 166 | } |
| 167 | |
Sen Jiang | 5288bd1 | 2018-05-03 11:39:04 -0700 | [diff] [blame^] | 168 | TYPED_TEST(ZipTest, CompressELFTest) { |
| 169 | string path = test_utils::GetBuildArtifactsPath("delta_generator"); |
| 170 | brillo::Blob in; |
| 171 | utils::ReadFile(path, &in); |
| 172 | brillo::Blob out; |
| 173 | EXPECT_TRUE(this->ZipCompress(in, &out)); |
| 174 | EXPECT_LT(out.size(), in.size()); |
| 175 | EXPECT_GT(out.size(), 0U); |
| 176 | brillo::Blob decompressed; |
| 177 | EXPECT_TRUE(this->ZipDecompress(out, &decompressed)); |
| 178 | EXPECT_EQ(in.size(), decompressed.size()); |
| 179 | EXPECT_EQ(0, memcmp(in.data(), decompressed.data(), in.size())); |
| 180 | } |
| 181 | |
Andrew de los Reyes | d2135f3 | 2010-03-11 16:00:28 -0800 | [diff] [blame] | 182 | } // namespace chromeos_update_engine |