|  | /* | 
|  | * 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 |