blob: a4572396e6519491229504694dc4a596ff80c734 [file] [log] [blame]
// Copyright 2015 PDFium 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 "core/fpdfapi/parser/fpdf_parser_decode.h"
#include "core/fpdfapi/parser/cpdf_array.h"
#include "core/fpdfapi/parser/cpdf_name.h"
#include "testing/gtest/include/gtest/gtest.h"
#include "testing/test_support.h"
TEST(fpdf_parser_decode, ValidateDecoderPipeline) {
{
// Empty decoder list is always valid.
CPDF_Array decoders;
EXPECT_TRUE(ValidateDecoderPipeline(&decoders));
}
{
// 1 decoder is always valid.
CPDF_Array decoders;
decoders.AddNew<CPDF_Name>("FlateEncode");
EXPECT_TRUE(ValidateDecoderPipeline(&decoders));
}
{
// 1 decoder is always valid, even with an unknown decoder.
CPDF_Array decoders;
decoders.AddNew<CPDF_Name>("FooBar");
EXPECT_TRUE(ValidateDecoderPipeline(&decoders));
}
{
// Valid 2 decoder pipeline.
CPDF_Array decoders;
decoders.AddNew<CPDF_Name>("AHx");
decoders.AddNew<CPDF_Name>("LZWDecode");
EXPECT_TRUE(ValidateDecoderPipeline(&decoders));
}
{
// Valid 2 decoder pipeline.
CPDF_Array decoders;
decoders.AddNew<CPDF_Name>("ASCII85Decode");
decoders.AddNew<CPDF_Name>("ASCII85Decode");
EXPECT_TRUE(ValidateDecoderPipeline(&decoders));
}
{
// Valid 5 decoder pipeline.
CPDF_Array decoders;
decoders.AddNew<CPDF_Name>("ASCII85Decode");
decoders.AddNew<CPDF_Name>("A85");
decoders.AddNew<CPDF_Name>("RunLengthDecode");
decoders.AddNew<CPDF_Name>("FlateDecode");
decoders.AddNew<CPDF_Name>("RL");
EXPECT_TRUE(ValidateDecoderPipeline(&decoders));
}
{
// Valid 5 decoder pipeline, with an image decoder at the end.
CPDF_Array decoders;
decoders.AddNew<CPDF_Name>("RunLengthDecode");
decoders.AddNew<CPDF_Name>("ASCII85Decode");
decoders.AddNew<CPDF_Name>("FlateDecode");
decoders.AddNew<CPDF_Name>("LZW");
decoders.AddNew<CPDF_Name>("DCTDecode");
EXPECT_TRUE(ValidateDecoderPipeline(&decoders));
}
{
// Invalid 2 decoder pipeline, with 2 image decoders.
CPDF_Array decoders;
decoders.AddNew<CPDF_Name>("DCTDecode");
decoders.AddNew<CPDF_Name>("CCITTFaxDecode");
EXPECT_FALSE(ValidateDecoderPipeline(&decoders));
}
{
// Invalid 2 decoder pipeline, with 1 image decoder at the start.
CPDF_Array decoders;
decoders.AddNew<CPDF_Name>("DCTDecode");
decoders.AddNew<CPDF_Name>("FlateDecode");
EXPECT_FALSE(ValidateDecoderPipeline(&decoders));
}
{
// Invalid 5 decoder pipeline.
CPDF_Array decoders;
decoders.AddNew<CPDF_Name>("FlateDecode");
decoders.AddNew<CPDF_Name>("FlateDecode");
decoders.AddNew<CPDF_Name>("DCTDecode");
decoders.AddNew<CPDF_Name>("FlateDecode");
decoders.AddNew<CPDF_Name>("FlateDecode");
EXPECT_FALSE(ValidateDecoderPipeline(&decoders));
}
}
TEST(fpdf_parser_decode, A85Decode) {
const pdfium::DecodeTestData test_data[] = {
// Empty src string.
STR_IN_OUT_CASE("", "", 0),
// Empty content in src string.
STR_IN_OUT_CASE("~>", "", 0),
// Regular conversion.
STR_IN_OUT_CASE("FCfN8~>", "test", 7),
// End at the ending mark.
STR_IN_OUT_CASE("FCfN8~>FCfN8", "test", 7),
// Skip whitespaces.
STR_IN_OUT_CASE("\t F C\r\n \tf N 8 ~>", "test", 17),
// No ending mark.
STR_IN_OUT_CASE("@3B0)DJj_BF*)>@Gp#-s", "a funny story :)", 20),
// Non-multiple length.
STR_IN_OUT_CASE("12A", "2k", 3),
// Stop at unknown characters.
STR_IN_OUT_CASE("FCfN8FCfN8vw", "testtest", 11),
};
for (size_t i = 0; i < FX_ArraySize(test_data); ++i) {
const pdfium::DecodeTestData* ptr = &test_data[i];
std::unique_ptr<uint8_t, FxFreeDeleter> result;
uint32_t result_size = 0;
EXPECT_EQ(ptr->processed_size,
A85Decode({ptr->input, ptr->input_size}, &result, &result_size))
<< "for case " << i;
ASSERT_EQ(ptr->expected_size, result_size);
const uint8_t* result_ptr = result.get();
for (size_t j = 0; j < result_size; ++j) {
EXPECT_EQ(ptr->expected[j], result_ptr[j])
<< "for case " << i << " char " << j;
}
}
}
TEST(fpdf_parser_decode, HexDecode) {
const pdfium::DecodeTestData test_data[] = {
// Empty src string.
STR_IN_OUT_CASE("", "", 0),
// Empty content in src string.
STR_IN_OUT_CASE(">", "", 1),
// Only whitespaces in src string.
STR_IN_OUT_CASE("\t \r\n>", "", 7),
// Regular conversion.
STR_IN_OUT_CASE("12Ac>zzz", "\x12\xac", 5),
// Skip whitespaces.
STR_IN_OUT_CASE("12 Ac\t02\r\nBF>zzz>", "\x12\xac\x02\xbf", 13),
// Non-multiple length.
STR_IN_OUT_CASE("12A>zzz", "\x12\xa0", 4),
// Skips unknown characters.
STR_IN_OUT_CASE("12tk \tAc>zzz", "\x12\xac", 10),
// No ending mark.
STR_IN_OUT_CASE("12AcED3c3456", "\x12\xac\xed\x3c\x34\x56", 12),
};
for (size_t i = 0; i < FX_ArraySize(test_data); ++i) {
const pdfium::DecodeTestData* ptr = &test_data[i];
std::unique_ptr<uint8_t, FxFreeDeleter> result;
uint32_t result_size = 0;
EXPECT_EQ(ptr->processed_size,
HexDecode({ptr->input, ptr->input_size}, &result, &result_size))
<< "for case " << i;
ASSERT_EQ(ptr->expected_size, result_size);
const uint8_t* result_ptr = result.get();
for (size_t j = 0; j < result_size; ++j) {
EXPECT_EQ(ptr->expected[j], result_ptr[j])
<< "for case " << i << " char " << j;
}
}
}
TEST(fpdf_parser_decode, EncodeText) {
struct EncodeTestData {
const wchar_t* input;
const char* expected_output;
size_t expected_length;
} test_data[] = {
// Empty src string.
{L"", "", 0},
// ASCII text.
{L"the quick\tfox", "the quick\tfox", 13},
// Unicode text.
{L"\x0330\x0331", "\xFE\xFF\x03\x30\x03\x31", 6},
// More Unicode text.
{L"\x7F51\x9875\x0020\x56FE\x7247\x0020"
L"\x8D44\x8BAF\x66F4\x591A\x0020\x00BB",
"\xFE\xFF\x7F\x51\x98\x75\x00\x20\x56\xFE\x72\x47\x00"
"\x20\x8D\x44\x8B\xAF\x66\xF4\x59\x1A\x00\x20\x00\xBB",
26},
};
for (size_t i = 0; i < FX_ArraySize(test_data); ++i) {
const auto& test_case = test_data[i];
ByteString output = PDF_EncodeText(test_case.input);
ASSERT_EQ(test_case.expected_length, output.GetLength()) << "for case "
<< i;
const char* str_ptr = output.c_str();
for (size_t j = 0; j < test_case.expected_length; ++j) {
EXPECT_EQ(test_case.expected_output[j], str_ptr[j]) << "for case " << i
<< " char " << j;
}
}
}