mirror of
https://git.suyu.dev/suyu/breakpad.git
synced 2025-12-31 03:37:44 +01:00
TYPED_TEST_CASE is deprecated in modern googletest. BUG=chromium:936654 Change-Id: I08004ffbb26089ebe17302934ed6d3268220d151 Reviewed-on: https://chromium-review.googlesource.com/c/1493423 Reviewed-by: Mark Mentovai <mark@chromium.org>
204 lines
7.3 KiB
C++
204 lines
7.3 KiB
C++
// Copyright (c) 2011 Google Inc.
|
|
// All rights reserved.
|
|
//
|
|
// Redistribution and use in source and binary forms, with or without
|
|
// modification, are permitted provided that the following conditions are
|
|
// met:
|
|
//
|
|
// * Redistributions of source code must retain the above copyright
|
|
// notice, this list of conditions and the following disclaimer.
|
|
// * Redistributions in binary form must reproduce the above
|
|
// copyright notice, this list of conditions and the following disclaimer
|
|
// in the documentation and/or other materials provided with the
|
|
// distribution.
|
|
// * Neither the name of Google Inc. nor the names of its
|
|
// contributors may be used to endorse or promote products derived from
|
|
// this software without specific prior written permission.
|
|
//
|
|
// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
|
|
// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
|
|
// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
|
|
// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
|
|
// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
|
// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
// Original author: Ted Mielczarek <ted.mielczarek@gmail.com>
|
|
|
|
// dump_symbols_unittest.cc:
|
|
// Unittests for google_breakpad::DumpSymbols
|
|
|
|
#include <elf.h>
|
|
#include <link.h>
|
|
#include <stdio.h>
|
|
|
|
#include <sstream>
|
|
#include <vector>
|
|
|
|
#include "breakpad_googletest_includes.h"
|
|
#include "common/linux/elf_gnu_compat.h"
|
|
#include "common/linux/elfutils.h"
|
|
#include "common/linux/dump_symbols.h"
|
|
#include "common/linux/synth_elf.h"
|
|
#include "common/module.h"
|
|
#include "common/using_std_string.h"
|
|
|
|
namespace google_breakpad {
|
|
|
|
bool ReadSymbolDataInternal(const uint8_t* obj_file,
|
|
const string& obj_filename,
|
|
const std::vector<string>& debug_dir,
|
|
const DumpOptions& options,
|
|
Module** module);
|
|
|
|
using google_breakpad::synth_elf::ELF;
|
|
using google_breakpad::synth_elf::Notes;
|
|
using google_breakpad::synth_elf::StringTable;
|
|
using google_breakpad::synth_elf::SymbolTable;
|
|
using google_breakpad::test_assembler::kLittleEndian;
|
|
using google_breakpad::test_assembler::Section;
|
|
using std::stringstream;
|
|
using std::vector;
|
|
using ::testing::Test;
|
|
using ::testing::Types;
|
|
|
|
template<typename ElfClass>
|
|
class DumpSymbols : public Test {
|
|
public:
|
|
void GetElfContents(ELF& elf) {
|
|
string contents;
|
|
ASSERT_TRUE(elf.GetContents(&contents));
|
|
ASSERT_LT(0U, contents.size());
|
|
|
|
elfdata_v.clear();
|
|
elfdata_v.insert(elfdata_v.begin(), contents.begin(), contents.end());
|
|
elfdata = &elfdata_v[0];
|
|
}
|
|
|
|
vector<uint8_t> elfdata_v;
|
|
uint8_t* elfdata;
|
|
};
|
|
|
|
typedef Types<ElfClass32, ElfClass64> ElfClasses;
|
|
|
|
TYPED_TEST_SUITE(DumpSymbols, ElfClasses);
|
|
|
|
TYPED_TEST(DumpSymbols, Invalid) {
|
|
Elf32_Ehdr header;
|
|
memset(&header, 0, sizeof(header));
|
|
Module* module;
|
|
DumpOptions options(ALL_SYMBOL_DATA, true);
|
|
EXPECT_FALSE(ReadSymbolDataInternal(reinterpret_cast<uint8_t*>(&header),
|
|
"foo",
|
|
vector<string>(),
|
|
options,
|
|
&module));
|
|
}
|
|
|
|
TYPED_TEST(DumpSymbols, SimplePublic) {
|
|
ELF elf(TypeParam::kMachine, TypeParam::kClass, kLittleEndian);
|
|
// Zero out text section for simplicity.
|
|
Section text(kLittleEndian);
|
|
text.Append(4096, 0);
|
|
elf.AddSection(".text", text, SHT_PROGBITS);
|
|
|
|
// Add a public symbol.
|
|
StringTable table(kLittleEndian);
|
|
SymbolTable syms(kLittleEndian, TypeParam::kAddrSize, table);
|
|
syms.AddSymbol("superfunc",
|
|
(typename TypeParam::Addr)0x1000,
|
|
(typename TypeParam::Addr)0x10,
|
|
// ELF32_ST_INFO works for 32-or 64-bit.
|
|
ELF32_ST_INFO(STB_GLOBAL, STT_FUNC),
|
|
SHN_UNDEF + 1);
|
|
int index = elf.AddSection(".dynstr", table, SHT_STRTAB);
|
|
elf.AddSection(".dynsym", syms,
|
|
SHT_DYNSYM, // type
|
|
SHF_ALLOC, // flags
|
|
0, // addr
|
|
index, // link
|
|
sizeof(typename TypeParam::Sym)); // entsize
|
|
|
|
elf.Finish();
|
|
this->GetElfContents(elf);
|
|
|
|
Module* module;
|
|
DumpOptions options(ALL_SYMBOL_DATA, true);
|
|
EXPECT_TRUE(ReadSymbolDataInternal(this->elfdata,
|
|
"foo",
|
|
vector<string>(),
|
|
options,
|
|
&module));
|
|
|
|
stringstream s;
|
|
module->Write(s, ALL_SYMBOL_DATA);
|
|
const string expected =
|
|
string("MODULE Linux ") + TypeParam::kMachineName
|
|
+ " 000000000000000000000000000000000 foo\n"
|
|
"INFO CODE_ID 00000000000000000000000000000000\n"
|
|
"PUBLIC 1000 0 superfunc\n";
|
|
EXPECT_EQ(expected, s.str());
|
|
delete module;
|
|
}
|
|
|
|
TYPED_TEST(DumpSymbols, SimpleBuildID) {
|
|
ELF elf(TypeParam::kMachine, TypeParam::kClass, kLittleEndian);
|
|
// Zero out text section for simplicity.
|
|
Section text(kLittleEndian);
|
|
text.Append(4096, 0);
|
|
elf.AddSection(".text", text, SHT_PROGBITS);
|
|
|
|
// Add a Build ID
|
|
const uint8_t kExpectedIdentifierBytes[] =
|
|
{0x00, 0x01, 0x02, 0x03, 0x04, 0x05, 0x06, 0x07,
|
|
0x08, 0x09, 0x0A, 0x0B, 0x0C, 0x0D, 0x0E, 0x0F,
|
|
0x10, 0x11, 0x12, 0x13};
|
|
Notes notes(kLittleEndian);
|
|
notes.AddNote(NT_GNU_BUILD_ID, "GNU", kExpectedIdentifierBytes,
|
|
sizeof(kExpectedIdentifierBytes));
|
|
elf.AddSection(".note.gnu.build-id", notes, SHT_NOTE);
|
|
|
|
// Add a public symbol.
|
|
StringTable table(kLittleEndian);
|
|
SymbolTable syms(kLittleEndian, TypeParam::kAddrSize, table);
|
|
syms.AddSymbol("superfunc",
|
|
(typename TypeParam::Addr)0x1000,
|
|
(typename TypeParam::Addr)0x10,
|
|
// ELF32_ST_INFO works for 32-or 64-bit.
|
|
ELF32_ST_INFO(STB_GLOBAL, STT_FUNC),
|
|
SHN_UNDEF + 1);
|
|
int index = elf.AddSection(".dynstr", table, SHT_STRTAB);
|
|
elf.AddSection(".dynsym", syms,
|
|
SHT_DYNSYM, // type
|
|
SHF_ALLOC, // flags
|
|
0, // addr
|
|
index, // link
|
|
sizeof(typename TypeParam::Sym)); // entsize
|
|
|
|
elf.Finish();
|
|
this->GetElfContents(elf);
|
|
|
|
Module* module;
|
|
DumpOptions options(ALL_SYMBOL_DATA, true);
|
|
EXPECT_TRUE(ReadSymbolDataInternal(this->elfdata,
|
|
"foo",
|
|
vector<string>(),
|
|
options,
|
|
&module));
|
|
|
|
stringstream s;
|
|
module->Write(s, ALL_SYMBOL_DATA);
|
|
const string expected =
|
|
string("MODULE Linux ") + TypeParam::kMachineName
|
|
+ " 030201000504070608090A0B0C0D0E0F0 foo\n"
|
|
"INFO CODE_ID 000102030405060708090A0B0C0D0E0F10111213\n"
|
|
"PUBLIC 1000 0 superfunc\n";
|
|
EXPECT_EQ(expected, s.str());
|
|
delete module;
|
|
}
|
|
|
|
} // namespace google_breakpad
|