| /* |
| * Copyright (C) 2017 The Android Open Source Project |
| * |
| * Licensed under the Apache License, Version 2.0 (the "License"); |
| * you may not use this file except in compliance with the License. |
| * You may obtain a copy of the License at |
| * |
| * http://www.apache.org/licenses/LICENSE-2.0 |
| * |
| * Unless required by applicable law or agreed to in writing, software |
| * distributed under the License is distributed on an "AS IS" BASIS, |
| * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. |
| * See the License for the specific language governing permissions and |
| * limitations under the License. |
| */ |
| |
| |
| #include "compact_dex_file.h" |
| #include "dex_file_loader.h" |
| #include "gtest/gtest.h" |
| |
| namespace art { |
| |
| TEST(CompactDexFileTest, MagicAndVersion) { |
| // Test permutations of valid/invalid headers. |
| for (size_t i = 0; i < 2; ++i) { |
| for (size_t j = 0; j < 2; ++j) { |
| static const size_t len = CompactDexFile::kDexVersionLen + CompactDexFile::kDexMagicSize; |
| uint8_t header[len] = {}; |
| std::fill_n(header, len, 0x99); |
| const bool valid_magic = (i & 1) == 0; |
| const bool valid_version = (j & 1) == 0; |
| if (valid_magic) { |
| CompactDexFile::WriteMagic(header); |
| } |
| if (valid_version) { |
| CompactDexFile::WriteCurrentVersion(header); |
| } |
| EXPECT_EQ(valid_magic, CompactDexFile::IsMagicValid(header)); |
| EXPECT_EQ(valid_version, CompactDexFile::IsVersionValid(header)); |
| EXPECT_EQ(valid_magic, DexFileLoader::IsMagicValid(header)); |
| EXPECT_EQ(valid_magic && valid_version, DexFileLoader::IsVersionAndMagicValid(header)); |
| } |
| } |
| } |
| |
| TEST(CompactDexFileTest, CodeItemFields) { |
| auto test_and_write = [&] (uint16_t registers_size, |
| uint16_t ins_size, |
| uint16_t outs_size, |
| uint16_t tries_size, |
| uint32_t insns_size_in_code_units) { |
| ASSERT_GE(registers_size, ins_size); |
| uint16_t buffer[sizeof(CompactDexFile::CodeItem) + |
| CompactDexFile::CodeItem::kMaxPreHeaderSize] = {}; |
| CompactDexFile::CodeItem* code_item = reinterpret_cast<CompactDexFile::CodeItem*>( |
| &buffer[CompactDexFile::CodeItem::kMaxPreHeaderSize]); |
| const uint16_t* preheader_ptr = code_item->Create(registers_size, |
| ins_size, |
| outs_size, |
| tries_size, |
| insns_size_in_code_units, |
| code_item->GetPreHeader()); |
| ASSERT_GT(preheader_ptr, buffer); |
| |
| uint16_t out_registers_size; |
| uint16_t out_ins_size; |
| uint16_t out_outs_size; |
| uint16_t out_tries_size; |
| uint32_t out_insns_size_in_code_units; |
| code_item->DecodeFields</*kDecodeOnlyInstructionCount*/false>(&out_insns_size_in_code_units, |
| &out_registers_size, |
| &out_ins_size, |
| &out_outs_size, |
| &out_tries_size); |
| ASSERT_EQ(registers_size, out_registers_size); |
| ASSERT_EQ(ins_size, out_ins_size); |
| ASSERT_EQ(outs_size, out_outs_size); |
| ASSERT_EQ(tries_size, out_tries_size); |
| ASSERT_EQ(insns_size_in_code_units, out_insns_size_in_code_units); |
| |
| ++out_insns_size_in_code_units; // Force value to change. |
| code_item->DecodeFields</*kDecodeOnlyInstructionCount*/true>(&out_insns_size_in_code_units, |
| /*registers_size*/ nullptr, |
| /*ins_size*/ nullptr, |
| /*outs_size*/ nullptr, |
| /*tries_size*/ nullptr); |
| ASSERT_EQ(insns_size_in_code_units, out_insns_size_in_code_units); |
| }; |
| static constexpr uint32_t kMax32 = std::numeric_limits<uint32_t>::max(); |
| static constexpr uint16_t kMax16 = std::numeric_limits<uint16_t>::max(); |
| test_and_write(0, 0, 0, 0, 0); |
| test_and_write(kMax16, kMax16, kMax16, kMax16, kMax32); |
| test_and_write(kMax16 - 1, kMax16 - 2, kMax16 - 3, kMax16 - 4, kMax32 - 5); |
| test_and_write(kMax16 - 4, kMax16 - 5, kMax16 - 3, kMax16 - 2, kMax32 - 1); |
| test_and_write(5, 4, 3, 2, 1); |
| test_and_write(5, 0, 3, 2, 1); |
| test_and_write(kMax16, 0, kMax16 / 2, 1234, kMax32 / 4); |
| } |
| |
| } // namespace art |