| /* |
| * Copyright 2019 Google Inc. |
| * |
| * Use of this source code is governed by a BSD-style license that can be |
| * found in the LICENSE file. |
| */ |
| |
| #include "include/core/SkData.h" |
| #include "include/core/SkRefCnt.h" |
| #include "include/core/SkTypes.h" |
| #include "src/core/SkDescriptor.h" |
| #include "src/core/SkReadBuffer.h" |
| #include "src/core/SkScalerContext.h" |
| #include "src/core/SkWriteBuffer.h" |
| #include "tests/Test.h" |
| |
| #include <cstddef> |
| #include <cstdint> |
| #include <memory> |
| #include <optional> |
| |
| class SkDescriptorTestHelper { |
| public: |
| static void SetLength(SkDescriptor* desc, size_t length) { desc->fLength = length; } |
| static void SetCount(SkDescriptor* desc, uint32_t count) { desc->fCount = count; } |
| }; |
| |
| DEF_TEST(Descriptor_empty, r) { |
| const size_t size = sizeof(SkDescriptor); |
| |
| auto desc = SkDescriptor::Alloc(size); |
| REPORTER_ASSERT(r, desc->isValid()); |
| REPORTER_ASSERT(r, desc->getLength() == size); |
| } |
| |
| DEF_TEST(Descriptor_valid_simple, r) { |
| const size_t size = |
| sizeof(SkDescriptor) + sizeof(SkDescriptor::Entry) + sizeof(SkScalerContextRec); |
| |
| auto desc = SkDescriptor::Alloc(size); |
| SkScalerContextRec rec; |
| desc->addEntry(kRec_SkDescriptorTag, sizeof(rec), &rec); |
| REPORTER_ASSERT(r, desc->isValid()); |
| REPORTER_ASSERT(r, desc->getLength() == size); |
| |
| SkDescriptorTestHelper::SetLength(desc.get(), size - 4); |
| REPORTER_ASSERT(r, !desc->isValid()); |
| } |
| |
| DEF_TEST(Descriptor_valid_simple_extra_space, r) { |
| const size_t extra_space = 100; |
| const size_t size = |
| sizeof(SkDescriptor) + sizeof(SkDescriptor::Entry) + sizeof(SkScalerContextRec); |
| |
| auto desc = SkDescriptor::Alloc(size + extra_space); |
| SkScalerContextRec rec; |
| desc->addEntry(kRec_SkDescriptorTag, sizeof(rec), &rec); |
| REPORTER_ASSERT(r, desc->isValid()); |
| REPORTER_ASSERT(r, desc->getLength() == size); |
| |
| SkDescriptorTestHelper::SetLength(desc.get(), size - 4); |
| REPORTER_ASSERT(r, !desc->isValid()); |
| } |
| |
| DEF_TEST(Descriptor_valid_more_tags, r) { |
| const size_t effectSize = 16; |
| const size_t testSize = 32; |
| const size_t size = sizeof(SkDescriptor) + 3 * sizeof(SkDescriptor::Entry) + |
| sizeof(SkScalerContextRec) + effectSize + testSize; |
| |
| auto desc = SkDescriptor::Alloc(size); |
| SkScalerContextRec rec; |
| desc->addEntry(kRec_SkDescriptorTag, sizeof(rec), &rec); |
| desc->addEntry(kEffects_SkDescriptorTag, effectSize, nullptr); |
| desc->addEntry(SkSetFourByteTag('t', 'e', 's', 't'), testSize, nullptr); |
| REPORTER_ASSERT(r, desc->isValid()); |
| REPORTER_ASSERT(r, desc->getLength() == size); |
| |
| SkDescriptorTestHelper::SetLength(desc.get(), size - 4); |
| REPORTER_ASSERT(r, !desc->isValid()); |
| } |
| |
| DEF_TEST(Descriptor_invalid_rec_size, r) { |
| const size_t size = |
| sizeof(SkDescriptor) + sizeof(SkDescriptor::Entry) + sizeof(SkScalerContextRec) - 4; |
| |
| auto desc = SkDescriptor::Alloc(size); |
| SkScalerContextRec rec; |
| desc->addEntry(kRec_SkDescriptorTag, sizeof(rec) - 4, &rec); |
| REPORTER_ASSERT(r, desc->getLength() == size); |
| REPORTER_ASSERT(r, !desc->isValid()); |
| } |
| |
| DEF_TEST(Descriptor_invalid_length, r) { |
| const size_t size = sizeof(SkDescriptor) + sizeof(SkDescriptor::Entry); |
| const size_t effect_size = 1000; |
| |
| auto desc = SkDescriptor::Alloc(size); |
| desc->addEntry(kEffects_SkDescriptorTag, effect_size, nullptr); |
| |
| SkDescriptorTestHelper::SetLength(desc.get(), size); |
| REPORTER_ASSERT(r, !desc->isValid()); |
| |
| SkDescriptorTestHelper::SetLength(desc.get(), size + effect_size); |
| REPORTER_ASSERT(r, desc->isValid()); |
| } |
| |
| DEF_TEST(Descriptor_entry_too_big, r) { |
| const size_t size = sizeof(SkDescriptor) + sizeof(SkDescriptor::Entry) + 4; |
| // Must be less than fLength, but big enough to be bigger then fLength when added. |
| const size_t effect_size = sizeof(SkDescriptor) + sizeof(SkDescriptor::Entry); |
| |
| auto desc = SkDescriptor::Alloc(size); |
| |
| desc->addEntry(kEffects_SkDescriptorTag, effect_size, nullptr); |
| |
| SkDescriptorTestHelper::SetLength(desc.get(), size); |
| SkDescriptorTestHelper::SetCount(desc.get(), 2); |
| REPORTER_ASSERT(r, !desc->isValid()); |
| |
| SkDescriptorTestHelper::SetLength(desc.get(), size); |
| SkDescriptorTestHelper::SetCount(desc.get(), 1); |
| REPORTER_ASSERT(r, !desc->isValid()); |
| } |
| |
| DEF_TEST(Descriptor_entry_over_end, r) { |
| auto desc = SkDescriptor::Alloc(36); |
| |
| // Make the start of the Entry be in the SkDescriptor, but the second half falls out side the |
| // SkDescriptor. So: 12 (for descriptor) + 8 (for entry) + 12 (for entry length) = 32. An |
| // An Entry is 8 bytes, so 4 bytes are < 36 and 4 bytes > 36. |
| desc->addEntry(kEffects_SkDescriptorTag, 12, nullptr); |
| |
| SkDescriptorTestHelper::SetLength(desc.get(), 36); |
| SkDescriptorTestHelper::SetCount(desc.get(), 2); |
| REPORTER_ASSERT(r, !desc->isValid()); |
| } |
| |
| DEF_TEST(Descriptor_flatten_unflatten, r) { |
| { |
| SkBinaryWriteBuffer writer({}); |
| auto desc = SkDescriptor::Alloc(sizeof(SkDescriptor)); |
| desc->computeChecksum(); |
| desc->flatten(writer); |
| auto data = writer.snapshotAsData(); |
| SkReadBuffer reader{data->data(), data->size()}; |
| auto ad = SkAutoDescriptor::MakeFromBuffer(reader); |
| REPORTER_ASSERT(r, ad.has_value()); |
| REPORTER_ASSERT(r, ad->getDesc()->isValid()); |
| } |
| |
| { // broken header |
| SkBinaryWriteBuffer writer({}); |
| writer.writeInt(0); // fChecksum |
| auto data = writer.snapshotAsData(); |
| SkReadBuffer reader{data->data(), data->size()}; |
| auto ad = SkAutoDescriptor::MakeFromBuffer(reader); |
| REPORTER_ASSERT(r, !ad.has_value()); |
| } |
| |
| { // length too big |
| SkBinaryWriteBuffer writer({}); |
| // Simulate a broken header |
| writer.writeInt(0); // fChecksum |
| writer.writeInt(4000); // fLength |
| writer.writeInt(0); // fCount |
| auto data = writer.snapshotAsData(); |
| SkReadBuffer reader{data->data(), data->size()}; |
| auto ad = SkAutoDescriptor::MakeFromBuffer(reader); |
| REPORTER_ASSERT(r, !ad.has_value()); |
| } |
| |
| { // length too small |
| SkBinaryWriteBuffer writer({}); |
| // Simulate a broken header |
| writer.writeInt(0); // fChecksum |
| writer.writeInt(3); // fLength |
| writer.writeInt(0); // fCount |
| auto data = writer.snapshotAsData(); |
| SkReadBuffer reader{data->data(), data->size()}; |
| auto ad = SkAutoDescriptor::MakeFromBuffer(reader); |
| REPORTER_ASSERT(r, !ad.has_value()); |
| } |
| |
| { // garbage in count |
| SkBinaryWriteBuffer writer({}); |
| // Simulate a broken header |
| writer.writeInt(0); // fChecksum |
| writer.writeInt(20); // fLength |
| writer.writeInt(10); // fCount |
| writer.writeInt(0); |
| writer.writeInt(0); |
| auto data = writer.snapshotAsData(); |
| SkReadBuffer reader{data->data(), data->size()}; |
| auto ad = SkAutoDescriptor::MakeFromBuffer(reader); |
| REPORTER_ASSERT(r, !ad.has_value()); |
| } |
| } |