| /* |
| * Copyright (C) 2016 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 <elf.h> |
| #include <errno.h> |
| #include <signal.h> |
| #include <string.h> |
| #include <sys/mman.h> |
| #include <sys/ptrace.h> |
| #include <sys/types.h> |
| #include <unistd.h> |
| |
| #include <memory> |
| #include <vector> |
| |
| #include <android-base/file.h> |
| #include <android-base/test_utils.h> |
| #include <gtest/gtest.h> |
| |
| #include <unwindstack/Elf.h> |
| #include <unwindstack/MapInfo.h> |
| #include <unwindstack/Memory.h> |
| |
| #include "ElfTestUtils.h" |
| |
| namespace unwindstack { |
| |
| class MapInfoGetElfTest : public ::testing::Test { |
| protected: |
| void SetUp() override { |
| map_ = mmap(nullptr, kMapSize, PROT_READ | PROT_WRITE, MAP_PRIVATE | MAP_ANONYMOUS, -1, 0); |
| ASSERT_NE(MAP_FAILED, map_); |
| |
| uint64_t start = reinterpret_cast<uint64_t>(map_); |
| info_.reset(new MapInfo{.start = start, .end = start + 1024, .offset = 0, .name = ""}); |
| } |
| |
| void TearDown() override { munmap(map_, kMapSize); } |
| |
| const size_t kMapSize = 4096; |
| |
| void* map_ = nullptr; |
| std::unique_ptr<MapInfo> info_; |
| }; |
| |
| TEST_F(MapInfoGetElfTest, invalid) { |
| // The map is empty, but this should still create an invalid elf object. |
| std::unique_ptr<Elf> elf(info_->GetElf(getpid(), false)); |
| ASSERT_TRUE(elf.get() != nullptr); |
| ASSERT_FALSE(elf->valid()); |
| } |
| |
| TEST_F(MapInfoGetElfTest, valid32) { |
| Elf32_Ehdr ehdr; |
| TestInitEhdr<Elf32_Ehdr>(&ehdr, ELFCLASS32, EM_ARM); |
| memcpy(map_, &ehdr, sizeof(ehdr)); |
| |
| std::unique_ptr<Elf> elf(info_->GetElf(getpid(), false)); |
| ASSERT_TRUE(elf.get() != nullptr); |
| ASSERT_TRUE(elf->valid()); |
| EXPECT_EQ(static_cast<uint32_t>(EM_ARM), elf->machine_type()); |
| EXPECT_EQ(ELFCLASS32, elf->class_type()); |
| } |
| |
| TEST_F(MapInfoGetElfTest, valid64) { |
| Elf64_Ehdr ehdr; |
| TestInitEhdr<Elf64_Ehdr>(&ehdr, ELFCLASS64, EM_AARCH64); |
| memcpy(map_, &ehdr, sizeof(ehdr)); |
| |
| std::unique_ptr<Elf> elf(info_->GetElf(getpid(), false)); |
| ASSERT_TRUE(elf.get() != nullptr); |
| ASSERT_TRUE(elf->valid()); |
| EXPECT_EQ(static_cast<uint32_t>(EM_AARCH64), elf->machine_type()); |
| EXPECT_EQ(ELFCLASS64, elf->class_type()); |
| } |
| |
| TEST_F(MapInfoGetElfTest, gnu_debugdata_do_not_init32) { |
| TestInitGnuDebugdata<Elf32_Ehdr, Elf32_Shdr>( |
| ELFCLASS32, EM_ARM, false, [&](uint64_t offset, const void* ptr, size_t size) { |
| memcpy(&reinterpret_cast<uint8_t*>(map_)[offset], ptr, size); |
| }); |
| |
| std::unique_ptr<Elf> elf(info_->GetElf(getpid(), false)); |
| ASSERT_TRUE(elf.get() != nullptr); |
| ASSERT_TRUE(elf->valid()); |
| EXPECT_EQ(static_cast<uint32_t>(EM_ARM), elf->machine_type()); |
| EXPECT_EQ(ELFCLASS32, elf->class_type()); |
| EXPECT_TRUE(elf->gnu_debugdata_interface() == nullptr); |
| } |
| |
| TEST_F(MapInfoGetElfTest, gnu_debugdata_do_not_init64) { |
| TestInitGnuDebugdata<Elf64_Ehdr, Elf64_Shdr>( |
| ELFCLASS64, EM_AARCH64, false, [&](uint64_t offset, const void* ptr, size_t size) { |
| memcpy(&reinterpret_cast<uint8_t*>(map_)[offset], ptr, size); |
| }); |
| |
| std::unique_ptr<Elf> elf(info_->GetElf(getpid(), false)); |
| ASSERT_TRUE(elf.get() != nullptr); |
| ASSERT_TRUE(elf->valid()); |
| EXPECT_EQ(static_cast<uint32_t>(EM_AARCH64), elf->machine_type()); |
| EXPECT_EQ(ELFCLASS64, elf->class_type()); |
| EXPECT_TRUE(elf->gnu_debugdata_interface() == nullptr); |
| } |
| |
| TEST_F(MapInfoGetElfTest, gnu_debugdata_init32) { |
| TestInitGnuDebugdata<Elf32_Ehdr, Elf32_Shdr>( |
| ELFCLASS32, EM_ARM, true, [&](uint64_t offset, const void* ptr, size_t size) { |
| memcpy(&reinterpret_cast<uint8_t*>(map_)[offset], ptr, size); |
| }); |
| |
| std::unique_ptr<Elf> elf(info_->GetElf(getpid(), true)); |
| ASSERT_TRUE(elf.get() != nullptr); |
| ASSERT_TRUE(elf->valid()); |
| EXPECT_EQ(static_cast<uint32_t>(EM_ARM), elf->machine_type()); |
| EXPECT_EQ(ELFCLASS32, elf->class_type()); |
| EXPECT_TRUE(elf->gnu_debugdata_interface() != nullptr); |
| } |
| |
| TEST_F(MapInfoGetElfTest, gnu_debugdata_init64) { |
| TestInitGnuDebugdata<Elf64_Ehdr, Elf64_Shdr>( |
| ELFCLASS64, EM_AARCH64, true, [&](uint64_t offset, const void* ptr, size_t size) { |
| memcpy(&reinterpret_cast<uint8_t*>(map_)[offset], ptr, size); |
| }); |
| |
| std::unique_ptr<Elf> elf(info_->GetElf(getpid(), true)); |
| ASSERT_TRUE(elf.get() != nullptr); |
| ASSERT_TRUE(elf->valid()); |
| EXPECT_EQ(static_cast<uint32_t>(EM_AARCH64), elf->machine_type()); |
| EXPECT_EQ(ELFCLASS64, elf->class_type()); |
| EXPECT_TRUE(elf->gnu_debugdata_interface() != nullptr); |
| } |
| |
| } // namespace unwindstack |