diff options
author | Tom Sepez <tsepez@chromium.org> | 2018-08-02 21:40:18 +0000 |
---|---|---|
committer | Chromium commit bot <commit-bot@chromium.org> | 2018-08-02 21:40:18 +0000 |
commit | 95340100f95f248defac47db8d63d6ce57b512d8 (patch) | |
tree | c83978ab7eb59bf85af228db4df5dc53ee8a2611 /core/fxcodec/jbig2/JBig2_Image_unittest.cpp | |
parent | 20e6688ab462d7ef749c1f97b83b5f325e88f698 (diff) | |
download | pdfium-95340100f95f248defac47db8d63d6ce57b512d8.tar.xz |
Tidy JBig2_Image.cppchromium/3511
Add checked/unchecked GetLine(y) methods and use them.
Introduce BIT_INDEX_TO_ALIGNED_BYTE() to de-mystify some shifting.
Move local declarations to spot of use.
Remove spurious Fill(), as we initialize to 0s.
Initialize members in header where possible.
Add unit tests.
Change-Id: I41ccb91b57320dbc790fd0f680f6d98571280343
Reviewed-on: https://pdfium-review.googlesource.com/39370
Commit-Queue: Tom Sepez <tsepez@chromium.org>
Reviewed-by: Lei Zhang <thestig@chromium.org>
Diffstat (limited to 'core/fxcodec/jbig2/JBig2_Image_unittest.cpp')
-rw-r--r-- | core/fxcodec/jbig2/JBig2_Image_unittest.cpp | 204 |
1 files changed, 199 insertions, 5 deletions
diff --git a/core/fxcodec/jbig2/JBig2_Image_unittest.cpp b/core/fxcodec/jbig2/JBig2_Image_unittest.cpp index bebfb6b20d..2b4d897cea 100644 --- a/core/fxcodec/jbig2/JBig2_Image_unittest.cpp +++ b/core/fxcodec/jbig2/JBig2_Image_unittest.cpp @@ -8,26 +8,75 @@ #include "core/fxcodec/jbig2/JBig2_Image.h" #include "testing/gtest/include/gtest/gtest.h" +#include "third_party/base/ptr_util.h" namespace { const int32_t kWidthPixels = 80; const int32_t kWidthBytes = 10; -const int32_t kStrideBytes = kWidthBytes + 1; // For testing stride != width. +const int32_t kStrideBytes = kWidthBytes + 2; // For testing stride != width. const int32_t kHeightLines = 20; const int32_t kLargerHeightLines = 100; const int32_t kTooLargeHeightLines = 40000000; +void CheckImageEq(CJBig2_Image* img1, CJBig2_Image* img2, int line) { + EXPECT_EQ(img1->width(), img2->width()); + EXPECT_EQ(img1->height(), img2->height()); + for (int32_t y = 0; y < img1->height(); ++y) { + for (int32_t x = 0; x < img1->width(); ++x) { + EXPECT_EQ(img1->GetPixel(x, y), img2->GetPixel(x, y)) + << " at " << x << " " << y << " actual line " << line; + } + } +} + } // namespace +TEST(fxcodec, EmptyImage) { + CJBig2_Image empty(0, 0); + EXPECT_EQ(empty.width(), 0); + EXPECT_EQ(empty.height(), 0); + + // Out-of-bounds SetPixel() is silent no-op. + empty.SetPixel(0, 0, true); + empty.SetPixel(1, 1, true); + + // Out-of-bounds GetPixel returns 0. + EXPECT_EQ(empty.GetPixel(0, 0), 0); + EXPECT_EQ(empty.GetPixel(1, 1), 0); + + // Out-of-bounds GetLine() returns null. + EXPECT_EQ(empty.GetLine(0), nullptr); + EXPECT_EQ(empty.GetLine(1), nullptr); +} + TEST(fxcodec, JBig2ImageCreate) { CJBig2_Image img(kWidthPixels, kHeightLines); - img.SetPixel(0, 0, true); - img.SetPixel(kWidthPixels - 1, kHeightLines - 1, false); EXPECT_EQ(kWidthPixels, img.width()); EXPECT_EQ(kHeightLines, img.height()); - EXPECT_TRUE(img.GetPixel(0, 0)); - EXPECT_FALSE(img.GetPixel(kWidthPixels - 1, kHeightLines - 1)); + EXPECT_EQ(0, img.GetPixel(0, 0)); + EXPECT_EQ(0, img.GetLine(0)[0]); + EXPECT_EQ(0, img.GetPixel(kWidthPixels - 1, kHeightLines - 1)); + EXPECT_EQ(0, img.GetLine(kHeightLines - 1)[kWidthBytes - 1]); + + img.SetPixel(0, 0, true); + img.SetPixel(kWidthPixels - 1, kHeightLines - 1, true); + EXPECT_EQ(1, img.GetPixel(0, 0)); + EXPECT_EQ(1, img.GetPixel(kWidthPixels - 1, kHeightLines - 1)); + EXPECT_EQ(0x80, img.GetLine(0)[0]); + EXPECT_EQ(0x01, img.GetLine(kHeightLines - 1)[kWidthBytes - 1]); + + // Out-of-bounds SetPixel() is silent no-op. + img.SetPixel(-1, 1, true); + img.SetPixel(kWidthPixels, kHeightLines, true); + + // Out-of-bounds GetPixel returns 0. + EXPECT_EQ(0, img.GetPixel(-1, -1)); + EXPECT_EQ(0, img.GetPixel(kWidthPixels, kHeightLines)); + + // Out-of-bounds GetLine() returns null. + EXPECT_EQ(nullptr, img.GetLine(-1)); + EXPECT_EQ(nullptr, img.GetLine(kHeightLines)); } TEST(fxcodec, JBig2ImageCreateTooBig) { @@ -56,6 +105,14 @@ TEST(fxcodec, JBig2ImageCreateExternalTooBig) { EXPECT_EQ(nullptr, img.data()); } +TEST(fxcodec, JBig2ImageCreateExternalBadStride) { + uint8_t buf[kHeightLines * kStrideBytes]; + CJBig2_Image img(kWidthPixels, kTooLargeHeightLines, kStrideBytes - 1, buf); + EXPECT_EQ(0, img.width()); + EXPECT_EQ(0, img.height()); + EXPECT_EQ(nullptr, img.data()); +} + TEST(fxcodec, JBig2ImageExpand) { CJBig2_Image img(kWidthPixels, kHeightLines); img.SetPixel(0, 0, true); @@ -103,3 +160,140 @@ TEST(fxcodec, JBig2ImageExpandExternalTooBig) { EXPECT_TRUE(img.GetPixel(0, 0)); EXPECT_FALSE(img.GetPixel(kWidthPixels - 1, kHeightLines - 1)); } + +TEST(fxcodec, JBig2EmptyImage) { + auto empty = pdfium::MakeUnique<CJBig2_Image>(0, 0); + + // Empty subimage. + auto sub1 = empty->SubImage(0, 0, 0, 0); + EXPECT_EQ(sub1->width(), 0); + EXPECT_EQ(sub1->height(), 0); + + // Larger dimensions are zero-padded. + auto sub2 = empty->SubImage(0, 0, 1, 1); + EXPECT_EQ(1, sub2->width()); + EXPECT_EQ(1, sub2->height()); + EXPECT_EQ(0, sub2->GetPixel(0, 0)); + + // Bad dimensions give an empty image. + sub2 = empty->SubImage(0, 0, -1, -1); + EXPECT_EQ(sub2->width(), 0); + EXPECT_EQ(sub2->height(), 0); + + // Bad offsets zero pad the image. + auto sub3 = empty->SubImage(-1, -1, 2, 2); + EXPECT_EQ(sub3->width(), 2); + EXPECT_EQ(sub3->height(), 2); + + // Bad dimensions and bad offsets give an empty image. + sub3 = empty->SubImage(-1, -1, -100, -100); + EXPECT_EQ(sub3->width(), 0); + EXPECT_EQ(sub3->height(), 0); +} + +TEST(fxcodec, JBig2SubImage) { + // 1-px wide rectangle in image. + uint8_t pattern[5][8] = { + {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, + {0x00, 0x01, 0xff, 0x80, 0x00, 0x00, 0x00, 0x00}, + {0x00, 0x01, 0x00, 0x80, 0x00, 0x00, 0x00, 0x00}, + {0x00, 0x01, 0xff, 0x80, 0x00, 0x00, 0x00, 0x00}, + {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, + }; + + // 1-px wide rectangle in image, offset 2 in x. + uint8_t pattern20[5][8] = { + {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, + {0x00, 0x07, 0xfe, 0x00, 0x00, 0x00, 0x00, 0x00}, + {0x00, 0x04, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00}, + {0x00, 0x07, 0xfe, 0x00, 0x00, 0x00, 0x00, 0x00}, + {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, + }; + + // 1-px wide rectangle in image, offset 2 in x and y, padded. + uint8_t pattern22[5][8] = { + {0x00, 0x04, 0x02, 0x00, 0x00, 0x00, 0x00, 0x00}, + {0x00, 0x07, 0xfe, 0x00, 0x00, 0x00, 0x00, 0x00}, + {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, + {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, + {0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00, 0x00}, + }; + + // Image size a nice clean power of two. + auto img32 = pdfium::MakeUnique<CJBig2_Image>( + 32, 5, 8, reinterpret_cast<uint8_t*>(pattern)); + + // Image size not a nice clean value. + auto img37 = pdfium::MakeUnique<CJBig2_Image>( + 37, 5, 8, reinterpret_cast<uint8_t*>(pattern)); + + // Expected results to check against. + auto expected20 = pdfium::MakeUnique<CJBig2_Image>( + 30, 5, 8, reinterpret_cast<uint8_t*>(pattern20)); + + auto expected22 = pdfium::MakeUnique<CJBig2_Image>( + 30, 5, 8, reinterpret_cast<uint8_t*>(pattern22)); + + auto expected_zeros = pdfium::MakeUnique<CJBig2_Image>(32, 5); + + // Empty subimage. + auto sub = img32->SubImage(0, 0, 0, 0); + EXPECT_EQ(sub->width(), 0); + EXPECT_EQ(sub->height(), 0); + + // Full sub-image. + sub = img32->SubImage(0, 0, 32, 5); + EXPECT_EQ(sub->width(), 32); + EXPECT_EQ(sub->height(), 5); + CheckImageEq(img32.get(), sub.get(), __LINE__); + + sub = img37->SubImage(0, 0, 32, 5); + EXPECT_EQ(sub->width(), 32); + EXPECT_EQ(sub->height(), 5); + CheckImageEq(img32.get(), sub.get(), __LINE__); + + // Actual bit manipulations. + sub = img32->SubImage(2, 0, 30, 5); + CheckImageEq(expected20.get(), sub.get(), __LINE__); + + sub = img37->SubImage(2, 2, 30, 5); + CheckImageEq(expected22.get(), sub.get(), __LINE__); + + // Aligned Sub-image including cruft in stride beyond width. + sub = img37->SubImage(32, 0, 32, 5); + CheckImageEq(expected_zeros.get(), sub.get(), __LINE__); + + // Sub-image waaaaay beyond width. + sub = img37->SubImage(2000, 0, 32, 5); + CheckImageEq(expected_zeros.get(), sub.get(), __LINE__); + + // Sub-image waaaaay beyond height. + sub = img37->SubImage(0, 2000, 32, 5); + CheckImageEq(expected_zeros.get(), sub.get(), __LINE__); + + // Sub-image with negative x offset. + sub = img37->SubImage(-1, 0, 32, 5); + CheckImageEq(expected_zeros.get(), sub.get(), __LINE__); + + // Sub-image with negative y offset. + sub = img37->SubImage(0, -1, 32, 5); + CheckImageEq(expected_zeros.get(), sub.get(), __LINE__); + + // Sub-image with negative width. + sub = img37->SubImage(-1, 0, 32, 5); + CheckImageEq(expected_zeros.get(), sub.get(), __LINE__); + + // Sub-image with negative height. + sub = img37->SubImage(0, -1, 32, 5); + CheckImageEq(expected_zeros.get(), sub.get(), __LINE__); + + // Sub-image wider than original. + sub = img37->SubImage(0, 0, 128, 5); + EXPECT_EQ(128, sub->width()); + EXPECT_EQ(5, sub->height()); + + // Sub-image higher than original. + sub = img37->SubImage(0, 0, 32, 40); + EXPECT_EQ(32, sub->width()); + EXPECT_EQ(40, sub->height()); +} |