| /* | 
 |  * 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 <stdint.h> | 
 |  | 
 | #include <gmock/gmock.h> | 
 | #include <gtest/gtest.h> | 
 |  | 
 | #include <unwindstack/DwarfSection.h> | 
 |  | 
 | #include "MemoryFake.h" | 
 |  | 
 | namespace unwindstack { | 
 |  | 
 | class MockDwarfSection : public DwarfSection { | 
 |  public: | 
 |   MockDwarfSection(Memory* memory) : DwarfSection(memory) {} | 
 |   virtual ~MockDwarfSection() = default; | 
 |  | 
 |   MOCK_METHOD(bool, Init, (uint64_t, uint64_t, int64_t), (override)); | 
 |  | 
 |   MOCK_METHOD(bool, Eval, (const DwarfCie*, Memory*, const dwarf_loc_regs_t&, Regs*, bool*), | 
 |               (override)); | 
 |  | 
 |   MOCK_METHOD(bool, Log, (uint8_t, uint64_t, const DwarfFde*), (override)); | 
 |  | 
 |   MOCK_METHOD(void, GetFdes, (std::vector<const DwarfFde*>*), (override)); | 
 |  | 
 |   MOCK_METHOD(const DwarfFde*, GetFdeFromPc, (uint64_t), (override)); | 
 |  | 
 |   MOCK_METHOD(bool, GetCfaLocationInfo, (uint64_t, const DwarfFde*, dwarf_loc_regs_t*), (override)); | 
 |  | 
 |   MOCK_METHOD(uint64_t, GetCieOffsetFromFde32, (uint32_t), (override)); | 
 |  | 
 |   MOCK_METHOD(uint64_t, GetCieOffsetFromFde64, (uint64_t), (override)); | 
 |  | 
 |   MOCK_METHOD(uint64_t, AdjustPcFromFde, (uint64_t), (override)); | 
 | }; | 
 |  | 
 | class DwarfSectionTest : public ::testing::Test { | 
 |  protected: | 
 |   void SetUp() override { section_.reset(new MockDwarfSection(&memory_)); } | 
 |  | 
 |   MemoryFake memory_; | 
 |   std::unique_ptr<MockDwarfSection> section_; | 
 | }; | 
 |  | 
 | TEST_F(DwarfSectionTest, Step_fail_fde) { | 
 |   EXPECT_CALL(*section_, GetFdeFromPc(0x1000)).WillOnce(::testing::Return(nullptr)); | 
 |  | 
 |   bool finished; | 
 |   ASSERT_FALSE(section_->Step(0x1000, nullptr, nullptr, &finished)); | 
 | } | 
 |  | 
 | TEST_F(DwarfSectionTest, Step_fail_cie_null) { | 
 |   DwarfFde fde{}; | 
 |   fde.pc_end = 0x2000; | 
 |   fde.cie = nullptr; | 
 |  | 
 |   EXPECT_CALL(*section_, GetFdeFromPc(0x1000)).WillOnce(::testing::Return(&fde)); | 
 |  | 
 |   bool finished; | 
 |   ASSERT_FALSE(section_->Step(0x1000, nullptr, nullptr, &finished)); | 
 | } | 
 |  | 
 | TEST_F(DwarfSectionTest, Step_fail_cfa_location) { | 
 |   DwarfCie cie{}; | 
 |   DwarfFde fde{}; | 
 |   fde.pc_end = 0x2000; | 
 |   fde.cie = &cie; | 
 |  | 
 |   EXPECT_CALL(*section_, GetFdeFromPc(0x1000)).WillOnce(::testing::Return(&fde)); | 
 |   EXPECT_CALL(*section_, GetCfaLocationInfo(0x1000, &fde, ::testing::_)) | 
 |       .WillOnce(::testing::Return(false)); | 
 |  | 
 |   bool finished; | 
 |   ASSERT_FALSE(section_->Step(0x1000, nullptr, nullptr, &finished)); | 
 | } | 
 |  | 
 | TEST_F(DwarfSectionTest, Step_pass) { | 
 |   DwarfCie cie{}; | 
 |   DwarfFde fde{}; | 
 |   fde.pc_end = 0x2000; | 
 |   fde.cie = &cie; | 
 |  | 
 |   EXPECT_CALL(*section_, GetFdeFromPc(0x1000)).WillOnce(::testing::Return(&fde)); | 
 |   EXPECT_CALL(*section_, GetCfaLocationInfo(0x1000, &fde, ::testing::_)) | 
 |       .WillOnce(::testing::Return(true)); | 
 |  | 
 |   MemoryFake process; | 
 |   EXPECT_CALL(*section_, Eval(&cie, &process, ::testing::_, nullptr, ::testing::_)) | 
 |       .WillOnce(::testing::Return(true)); | 
 |  | 
 |   bool finished; | 
 |   ASSERT_TRUE(section_->Step(0x1000, nullptr, &process, &finished)); | 
 | } | 
 |  | 
 | static bool MockGetCfaLocationInfo(::testing::Unused, const DwarfFde* fde, | 
 |                                    dwarf_loc_regs_t* loc_regs) { | 
 |   loc_regs->pc_start = fde->pc_start; | 
 |   loc_regs->pc_end = fde->pc_end; | 
 |   return true; | 
 | } | 
 |  | 
 | TEST_F(DwarfSectionTest, Step_cache) { | 
 |   DwarfCie cie{}; | 
 |   DwarfFde fde{}; | 
 |   fde.pc_start = 0x500; | 
 |   fde.pc_end = 0x2000; | 
 |   fde.cie = &cie; | 
 |  | 
 |   EXPECT_CALL(*section_, GetFdeFromPc(0x1000)).WillOnce(::testing::Return(&fde)); | 
 |   EXPECT_CALL(*section_, GetCfaLocationInfo(0x1000, &fde, ::testing::_)) | 
 |       .WillOnce(::testing::Invoke(MockGetCfaLocationInfo)); | 
 |  | 
 |   MemoryFake process; | 
 |   EXPECT_CALL(*section_, Eval(&cie, &process, ::testing::_, nullptr, ::testing::_)) | 
 |       .WillRepeatedly(::testing::Return(true)); | 
 |  | 
 |   bool finished; | 
 |   ASSERT_TRUE(section_->Step(0x1000, nullptr, &process, &finished)); | 
 |   ASSERT_TRUE(section_->Step(0x1000, nullptr, &process, &finished)); | 
 |   ASSERT_TRUE(section_->Step(0x1500, nullptr, &process, &finished)); | 
 | } | 
 |  | 
 | TEST_F(DwarfSectionTest, Step_cache_not_in_pc) { | 
 |   DwarfCie cie{}; | 
 |   DwarfFde fde0{}; | 
 |   fde0.pc_start = 0x1000; | 
 |   fde0.pc_end = 0x2000; | 
 |   fde0.cie = &cie; | 
 |   EXPECT_CALL(*section_, GetFdeFromPc(0x1000)).WillOnce(::testing::Return(&fde0)); | 
 |   EXPECT_CALL(*section_, GetCfaLocationInfo(0x1000, &fde0, ::testing::_)) | 
 |       .WillOnce(::testing::Invoke(MockGetCfaLocationInfo)); | 
 |  | 
 |   MemoryFake process; | 
 |   EXPECT_CALL(*section_, Eval(&cie, &process, ::testing::_, nullptr, ::testing::_)) | 
 |       .WillRepeatedly(::testing::Return(true)); | 
 |  | 
 |   bool finished; | 
 |   ASSERT_TRUE(section_->Step(0x1000, nullptr, &process, &finished)); | 
 |  | 
 |   DwarfFde fde1{}; | 
 |   fde1.pc_start = 0x500; | 
 |   fde1.pc_end = 0x800; | 
 |   fde1.cie = &cie; | 
 |   EXPECT_CALL(*section_, GetFdeFromPc(0x600)).WillOnce(::testing::Return(&fde1)); | 
 |   EXPECT_CALL(*section_, GetCfaLocationInfo(0x600, &fde1, ::testing::_)) | 
 |       .WillOnce(::testing::Invoke(MockGetCfaLocationInfo)); | 
 |  | 
 |   ASSERT_TRUE(section_->Step(0x600, nullptr, &process, &finished)); | 
 |   ASSERT_TRUE(section_->Step(0x700, nullptr, &process, &finished)); | 
 | } | 
 |  | 
 | }  // namespace unwindstack |