Index: chrome/utility/image_decoder_impl_unittest.cc |
diff --git a/chrome/utility/image_decoder_impl_unittest.cc b/chrome/utility/image_decoder_impl_unittest.cc |
new file mode 100644 |
index 0000000000000000000000000000000000000000..45f257c8699d5258d85c293186315bbe882d7369 |
--- /dev/null |
+++ b/chrome/utility/image_decoder_impl_unittest.cc |
@@ -0,0 +1,122 @@ |
+// Copyright 2016 The Chromium Authors. All rights reserved. |
+// Use of this source code is governed by a BSD-style license that can be |
+// found in the LICENSE file. |
+ |
+#include "chrome/utility/image_decoder_impl.h" |
+ |
+#include <vector> |
+ |
+#include "base/bind.h" |
+#include "ipc/ipc_channel.h" |
+#include "skia/public/type_converters.h" |
+#include "testing/gtest/include/gtest/gtest.h" |
+#include "third_party/skia/include/core/SkBitmap.h" |
+#include "ui/gfx/codec/jpeg_codec.h" |
+ |
+namespace mojom { |
+ |
+namespace { |
+ |
+bool CreateJPEGImage(int width, |
+ int height, |
+ SkColor color, |
+ std::vector<unsigned char>* output) { |
+ SkBitmap bitmap; |
+ bitmap.allocN32Pixels(width, height); |
+ bitmap.eraseColor(color); |
+ |
+ const int kQuality = 50; |
+ if (!gfx::JPEGCodec::Encode( |
+ static_cast<const unsigned char*>(bitmap.getPixels()), |
+ gfx::JPEGCodec::FORMAT_SkBitmap, width, height, |
+ static_cast<int>(bitmap.rowBytes()), kQuality, output)) { |
+ LOG(ERROR) << "Unable to encode " << width << "x" << height << " bitmap"; |
+ return false; |
+ } |
+ return true; |
+} |
+ |
+class Request { |
+ public: |
+ explicit Request(ImageDecoderImpl* decoder) : decoder_(decoder) {} |
+ |
+ void DecodeImage(const std::vector<unsigned char>& image, bool shrink) { |
+ decoder_->DecodeImage( |
+ mojo::Array<uint8_t>::From(image), ImageCodec::DEFAULT, |
+ shrink, base::Bind(&Request::OnRequestDone, base::Unretained(this))); |
+ } |
+ |
+ const skia::mojom::BitmapPtr& bitmap() const { return bitmap_; } |
+ |
+ private: |
+ void OnRequestDone(skia::mojom::BitmapPtr result_image) { |
+ bitmap_ = std::move(result_image); |
+ } |
+ |
+ ImageDecoderImpl* decoder_; |
+ skia::mojom::BitmapPtr bitmap_; |
+}; |
+ |
+} // namespace |
+ |
+// Test that DecodeImage() doesn't return image message > (max message size) |
+TEST(ImageDecoderImplTest, DecodeImageSizeLimit) { |
+ // Using actual limit generates 14000 x 9400 images, which causes the test to |
+ // timeout. We test with a smaller limit for efficiency. |
+ const size_t kTestMessageSize = IPC::Channel::kMaximumMessageSize / 1024; |
+ |
+ ImageDecoderImpl decoder(kTestMessageSize); |
+ |
+ // Approx max height for 3:2 image that will fit in IPC message; |
+ // 1.5 for width/height ratio, 4 for bytes/pixel |
+ int max_height_for_msg = sqrt(kTestMessageSize / (1.5 * 4)); |
+ int base_msg_size = sizeof(skia::mojom::Bitmap::Data_); |
+ |
+ // Sizes which should trigger dimension-halving 0, 1 and 2 times |
+ int heights[] = {max_height_for_msg - 10, |
+ max_height_for_msg + 10, |
+ 2 * max_height_for_msg + 10}; |
+ int widths[] = {heights[0] * 3 / 2, heights[1] * 3 / 2, heights[2] * 3 / 2}; |
+ for (size_t i = 0; i < arraysize(heights); i++) { |
+ std::vector<unsigned char> jpg; |
+ ASSERT_TRUE(CreateJPEGImage(widths[i], heights[i], SK_ColorRED, &jpg)); |
+ |
+ Request request(&decoder); |
+ request.DecodeImage(jpg, true); |
+ ASSERT_FALSE(request.bitmap().is_null()); |
+ SkBitmap bitmap = request.bitmap().To<SkBitmap>(); |
+ |
+ // Check that image has been shrunk appropriately |
+ EXPECT_LT(bitmap.computeSize64() + base_msg_size, |
+ static_cast<int64_t>(kTestMessageSize)); |
+// Android does its own image shrinking for memory conservation deeper in |
+// the decode, so more specific tests here won't work. |
+#if !defined(OS_ANDROID) |
+ EXPECT_EQ(widths[i] >> i, bitmap.width()); |
+ EXPECT_EQ(heights[i] >> i, bitmap.height()); |
+ |
+ // Check that if resize not requested and image exceeds IPC size limit, |
+ // an empty image is returned |
+ if (heights[i] > max_height_for_msg) { |
+ Request request(&decoder); |
+ request.DecodeImage(jpg, false); |
+ EXPECT_TRUE(request.bitmap().is_null()); |
+ } |
+#endif |
+ } |
+} |
+ |
+TEST(ImageDecoderImplTest, DecodeImageFailed) { |
+ ImageDecoderImpl decoder(IPC::Channel::kMaximumMessageSize); |
+ |
+ // The "jpeg" is just some "random" data; |
+ const char kRandomData[] = "u gycfy7xdjkhfgui bdui "; |
+ std::vector<unsigned char> jpg(kRandomData, |
+ kRandomData + sizeof(kRandomData)); |
+ |
+ Request request(&decoder); |
+ request.DecodeImage(jpg, false); |
+ EXPECT_TRUE(request.bitmap().is_null()); |
+} |
+ |
+} // namespace mojom |