mirror of
https://github.com/yuzu-emu/breakpad.git
synced 2025-07-06 12:40:42 +00:00
The Rust compiler uses GCC C++ name mangling, but it has another layer of encoding so abi::cxa_demangle doesn't produce great results. This patch changes dump_syms to dump unmangled names by default so that consumers can demangle them after-the-fact. It also adds a tiny bit of support for linking against a Rust library I wrote that can demangle Rust symbols nicely: https://github.com/luser/rust-demangle-capi BUG= Change-Id: I63a425035ebb7ac516f067fed2aa782849ea9604 Reviewed-on: https://chromium-review.googlesource.com/402308 Reviewed-by: Mark Mentovai <mark@chromium.org>
1853 lines
69 KiB
C++
1853 lines
69 KiB
C++
// Copyright (c) 2010 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: Jim Blandy <jimb@mozilla.com> <jimb@red-bean.com>
|
|
|
|
// dwarf_cu_to_module.cc: Unit tests for google_breakpad::DwarfCUToModule.
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <string>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
#include "breakpad_googletest_includes.h"
|
|
#include "common/dwarf_cu_to_module.h"
|
|
#include "common/using_std_string.h"
|
|
|
|
using std::make_pair;
|
|
using std::vector;
|
|
|
|
using dwarf2reader::DIEHandler;
|
|
using dwarf2reader::DwarfTag;
|
|
using dwarf2reader::DwarfAttribute;
|
|
using dwarf2reader::DwarfForm;
|
|
using dwarf2reader::DwarfInline;
|
|
using dwarf2reader::RootDIEHandler;
|
|
using google_breakpad::DwarfCUToModule;
|
|
using google_breakpad::Module;
|
|
|
|
using ::testing::_;
|
|
using ::testing::AtMost;
|
|
using ::testing::Invoke;
|
|
using ::testing::Return;
|
|
using ::testing::Test;
|
|
using ::testing::TestWithParam;
|
|
using ::testing::Values;
|
|
using ::testing::ValuesIn;
|
|
|
|
// Mock classes.
|
|
|
|
class MockLineToModuleHandler: public DwarfCUToModule::LineToModuleHandler {
|
|
public:
|
|
MOCK_METHOD1(StartCompilationUnit, void(const string& compilation_dir));
|
|
MOCK_METHOD4(ReadProgram, void(const uint8_t *program, uint64 length,
|
|
Module *module, vector<Module::Line> *lines));
|
|
};
|
|
|
|
class MockWarningReporter: public DwarfCUToModule::WarningReporter {
|
|
public:
|
|
MockWarningReporter(const string &filename, uint64 cu_offset)
|
|
: DwarfCUToModule::WarningReporter(filename, cu_offset) { }
|
|
MOCK_METHOD1(SetCUName, void(const string &name));
|
|
MOCK_METHOD2(UnknownSpecification, void(uint64 offset, uint64 target));
|
|
MOCK_METHOD2(UnknownAbstractOrigin, void(uint64 offset, uint64 target));
|
|
MOCK_METHOD1(MissingSection, void(const string §ion_name));
|
|
MOCK_METHOD1(BadLineInfoOffset, void(uint64 offset));
|
|
MOCK_METHOD1(UncoveredFunction, void(const Module::Function &function));
|
|
MOCK_METHOD1(UncoveredLine, void(const Module::Line &line));
|
|
MOCK_METHOD1(UnnamedFunction, void(uint64 offset));
|
|
MOCK_METHOD1(DemangleError, void(const string &input));
|
|
MOCK_METHOD2(UnhandledInterCUReference, void(uint64 offset, uint64 target));
|
|
};
|
|
|
|
// A fixture class including all the objects needed to handle a
|
|
// compilation unit, and their entourage. It includes member functions
|
|
// for doing common kinds of setup and tests.
|
|
class CUFixtureBase {
|
|
public:
|
|
// If we have:
|
|
//
|
|
// vector<Module::Line> lines;
|
|
// AppendLinesFunctor appender(lines);
|
|
//
|
|
// then doing:
|
|
//
|
|
// appender(line_program, length, module, line_vector);
|
|
//
|
|
// will append lines to the end of line_vector. We can use this with
|
|
// MockLineToModuleHandler like this:
|
|
//
|
|
// MockLineToModuleHandler l2m;
|
|
// EXPECT_CALL(l2m, ReadProgram(_,_,_,_))
|
|
// .WillOnce(DoAll(Invoke(appender), Return()));
|
|
//
|
|
// in which case calling l2m with some line vector will append lines.
|
|
class AppendLinesFunctor {
|
|
public:
|
|
explicit AppendLinesFunctor(
|
|
const vector<Module::Line> *lines) : lines_(lines) { }
|
|
void operator()(const uint8_t *program, uint64 length,
|
|
Module *module, vector<Module::Line> *lines) {
|
|
lines->insert(lines->end(), lines_->begin(), lines_->end());
|
|
}
|
|
private:
|
|
const vector<Module::Line> *lines_;
|
|
};
|
|
|
|
CUFixtureBase()
|
|
: module_("module-name", "module-os", "module-arch", "module-id"),
|
|
file_context_("dwarf-filename", &module_, true),
|
|
language_(dwarf2reader::DW_LANG_none),
|
|
language_signed_(false),
|
|
appender_(&lines_),
|
|
reporter_("dwarf-filename", 0xcf8f9bb6443d29b5LL),
|
|
root_handler_(&file_context_, &line_reader_, &reporter_),
|
|
functions_filled_(false) {
|
|
// By default, expect no warnings to be reported, and expect the
|
|
// compilation unit's name to be provided. The test can override
|
|
// these expectations.
|
|
EXPECT_CALL(reporter_, SetCUName("compilation-unit-name")).Times(1);
|
|
EXPECT_CALL(reporter_, UnknownSpecification(_, _)).Times(0);
|
|
EXPECT_CALL(reporter_, UnknownAbstractOrigin(_, _)).Times(0);
|
|
EXPECT_CALL(reporter_, MissingSection(_)).Times(0);
|
|
EXPECT_CALL(reporter_, BadLineInfoOffset(_)).Times(0);
|
|
EXPECT_CALL(reporter_, UncoveredFunction(_)).Times(0);
|
|
EXPECT_CALL(reporter_, UncoveredLine(_)).Times(0);
|
|
EXPECT_CALL(reporter_, UnnamedFunction(_)).Times(0);
|
|
EXPECT_CALL(reporter_, UnhandledInterCUReference(_, _)).Times(0);
|
|
|
|
// By default, expect the line program reader not to be invoked. We
|
|
// may override this in StartCU.
|
|
EXPECT_CALL(line_reader_, StartCompilationUnit(_)).Times(0);
|
|
EXPECT_CALL(line_reader_, ReadProgram(_,_,_,_)).Times(0);
|
|
|
|
// The handler will consult this section map to decide what to
|
|
// pass to our line reader.
|
|
file_context_.AddSectionToSectionMap(".debug_line",
|
|
dummy_line_program_,
|
|
dummy_line_size_);
|
|
}
|
|
|
|
// Add a line with the given address, size, filename, and line
|
|
// number to the end of the statement list the handler will receive
|
|
// when it invokes its LineToModuleHandler. Call this before calling
|
|
// StartCU.
|
|
void PushLine(Module::Address address, Module::Address size,
|
|
const string &filename, int line_number);
|
|
|
|
// Use LANGUAGE for the compilation unit. More precisely, arrange
|
|
// for StartCU to pass the compilation unit's root DIE a
|
|
// DW_AT_language attribute whose value is LANGUAGE.
|
|
void SetLanguage(dwarf2reader::DwarfLanguage language) {
|
|
language_ = language;
|
|
}
|
|
|
|
// If SIGNED true, have StartCU report DW_AT_language as a signed
|
|
// attribute; if false, have it report it as unsigned.
|
|
void SetLanguageSigned(bool is_signed) { language_signed_ = is_signed; }
|
|
|
|
// Call the handler this.root_handler_'s StartCompilationUnit and
|
|
// StartRootDIE member functions, passing it appropriate attributes as
|
|
// determined by prior calls to PushLine and SetLanguage. Leave
|
|
// this.root_handler_ ready to hear about children: call
|
|
// this.root_handler_.EndAttributes, but not this.root_handler_.Finish.
|
|
void StartCU();
|
|
|
|
// Have HANDLER process some strange attribute/form/value triples.
|
|
void ProcessStrangeAttributes(dwarf2reader::DIEHandler *handler);
|
|
|
|
// Start a child DIE of PARENT with the given tag and name. Leave
|
|
// the handler ready to hear about children: call EndAttributes, but
|
|
// not Finish.
|
|
DIEHandler *StartNamedDIE(DIEHandler *parent, DwarfTag tag,
|
|
const string &name);
|
|
|
|
// Start a child DIE of PARENT with the given tag and a
|
|
// DW_AT_specification attribute whose value is SPECIFICATION. Leave
|
|
// the handler ready to hear about children: call EndAttributes, but
|
|
// not Finish. If NAME is non-zero, use it as the DW_AT_name
|
|
// attribute.
|
|
DIEHandler *StartSpecifiedDIE(DIEHandler *parent, DwarfTag tag,
|
|
uint64 specification, const char *name = NULL);
|
|
|
|
// Define a function as a child of PARENT with the given name, address, and
|
|
// size. If high_pc_form is DW_FORM_addr then the DW_AT_high_pc attribute
|
|
// will be written as an address; otherwise it will be written as the
|
|
// function's size. Call EndAttributes and Finish; one cannot define
|
|
// children of the defined function's DIE.
|
|
void DefineFunction(DIEHandler *parent, const string &name,
|
|
Module::Address address, Module::Address size,
|
|
const char* mangled_name,
|
|
DwarfForm high_pc_form = dwarf2reader::DW_FORM_addr);
|
|
|
|
// Create a declaration DIE as a child of PARENT with the given
|
|
// offset, tag and name. If NAME is the empty string, don't provide
|
|
// a DW_AT_name attribute. Call EndAttributes and Finish.
|
|
void DeclarationDIE(DIEHandler *parent, uint64 offset,
|
|
DwarfTag tag, const string &name,
|
|
const string &mangled_name);
|
|
|
|
// Create a definition DIE as a child of PARENT with the given tag
|
|
// that refers to the declaration DIE at offset SPECIFICATION as its
|
|
// specification. If NAME is non-empty, pass it as the DW_AT_name
|
|
// attribute. If SIZE is non-zero, record ADDRESS and SIZE as
|
|
// low_pc/high_pc attributes.
|
|
void DefinitionDIE(DIEHandler *parent, DwarfTag tag,
|
|
uint64 specification, const string &name,
|
|
Module::Address address = 0, Module::Address size = 0);
|
|
|
|
// Create an inline DW_TAG_subprogram DIE as a child of PARENT. If
|
|
// SPECIFICATION is non-zero, then the DIE refers to the declaration DIE at
|
|
// offset SPECIFICATION as its specification. If Name is non-empty, pass it
|
|
// as the DW_AT_name attribute.
|
|
void AbstractInstanceDIE(DIEHandler *parent, uint64 offset,
|
|
DwarfInline type, uint64 specification,
|
|
const string &name,
|
|
DwarfForm form = dwarf2reader::DW_FORM_data1);
|
|
|
|
// Create a DW_TAG_subprogram DIE as a child of PARENT that refers to
|
|
// ORIGIN in its DW_AT_abstract_origin attribute. If NAME is the empty
|
|
// string, don't provide a DW_AT_name attribute.
|
|
void DefineInlineInstanceDIE(DIEHandler *parent, const string &name,
|
|
uint64 origin, Module::Address address,
|
|
Module::Address size);
|
|
|
|
// The following Test* functions should be called after calling
|
|
// this.root_handler_.Finish. After that point, no further calls
|
|
// should be made on the handler.
|
|
|
|
// Test that the number of functions defined in the module this.module_ is
|
|
// equal to EXPECTED.
|
|
void TestFunctionCount(size_t expected);
|
|
|
|
// Test that the I'th function (ordered by address) in the module
|
|
// this.module_ has the given name, address, and size, and that its
|
|
// parameter size is zero.
|
|
void TestFunction(int i, const string &name,
|
|
Module::Address address, Module::Address size);
|
|
|
|
// Test that the number of source lines owned by the I'th function
|
|
// in the module this.module_ is equal to EXPECTED.
|
|
void TestLineCount(int i, size_t expected);
|
|
|
|
// Test that the J'th line (ordered by address) of the I'th function
|
|
// (again, by address) has the given address, size, filename, and
|
|
// line number.
|
|
void TestLine(int i, int j, Module::Address address, Module::Address size,
|
|
const string &filename, int number);
|
|
|
|
// Actual objects under test.
|
|
Module module_;
|
|
DwarfCUToModule::FileContext file_context_;
|
|
|
|
// If this is not DW_LANG_none, we'll pass it as a DW_AT_language
|
|
// attribute to the compilation unit. This defaults to DW_LANG_none.
|
|
dwarf2reader::DwarfLanguage language_;
|
|
|
|
// If this is true, report DW_AT_language as a signed value; if false,
|
|
// report it as an unsigned value.
|
|
bool language_signed_;
|
|
|
|
// If this is not empty, we'll give the CU a DW_AT_comp_dir attribute that
|
|
// indicates the path that this compilation unit was compiled in.
|
|
string compilation_dir_;
|
|
|
|
// If this is not empty, we'll give the CU a DW_AT_stmt_list
|
|
// attribute that, when passed to line_reader_, adds these lines to the
|
|
// provided lines array.
|
|
vector<Module::Line> lines_;
|
|
|
|
// Mock line program reader.
|
|
MockLineToModuleHandler line_reader_;
|
|
AppendLinesFunctor appender_;
|
|
static const uint8_t dummy_line_program_[];
|
|
static const size_t dummy_line_size_;
|
|
|
|
MockWarningReporter reporter_;
|
|
DwarfCUToModule root_handler_;
|
|
|
|
private:
|
|
// Fill functions_, if we haven't already.
|
|
void FillFunctions();
|
|
|
|
// If functions_filled_ is true, this is a table of functions we've
|
|
// extracted from module_, sorted by address.
|
|
vector<Module::Function *> functions_;
|
|
// True if we have filled the above vector with this.module_'s function list.
|
|
bool functions_filled_;
|
|
};
|
|
|
|
const uint8_t CUFixtureBase::dummy_line_program_[] = "lots of fun data";
|
|
const size_t CUFixtureBase::dummy_line_size_ =
|
|
sizeof(CUFixtureBase::dummy_line_program_);
|
|
|
|
void CUFixtureBase::PushLine(Module::Address address, Module::Address size,
|
|
const string &filename, int line_number) {
|
|
Module::Line l;
|
|
l.address = address;
|
|
l.size = size;
|
|
l.file = module_.FindFile(filename);
|
|
l.number = line_number;
|
|
lines_.push_back(l);
|
|
}
|
|
|
|
void CUFixtureBase::StartCU() {
|
|
if (!compilation_dir_.empty())
|
|
EXPECT_CALL(line_reader_,
|
|
StartCompilationUnit(compilation_dir_)).Times(1);
|
|
|
|
// If we have lines, make the line reader expect to be invoked at
|
|
// most once. (Hey, if the handler can pass its tests without
|
|
// bothering to read the line number data, that's great.)
|
|
// Have it add the lines passed to PushLine. Otherwise, leave the
|
|
// initial expectation (no calls) in force.
|
|
if (!lines_.empty())
|
|
EXPECT_CALL(line_reader_,
|
|
ReadProgram(&dummy_line_program_[0], dummy_line_size_,
|
|
&module_, _))
|
|
.Times(AtMost(1))
|
|
.WillOnce(DoAll(Invoke(appender_), Return()));
|
|
|
|
ASSERT_TRUE(root_handler_
|
|
.StartCompilationUnit(0x51182ec307610b51ULL, 0x81, 0x44,
|
|
0x4241b4f33720dd5cULL, 3));
|
|
{
|
|
ASSERT_TRUE(root_handler_.StartRootDIE(0x02e56bfbda9e7337ULL,
|
|
dwarf2reader::DW_TAG_compile_unit));
|
|
}
|
|
root_handler_.ProcessAttributeString(dwarf2reader::DW_AT_name,
|
|
dwarf2reader::DW_FORM_strp,
|
|
"compilation-unit-name");
|
|
if (!compilation_dir_.empty())
|
|
root_handler_.ProcessAttributeString(dwarf2reader::DW_AT_comp_dir,
|
|
dwarf2reader::DW_FORM_strp,
|
|
compilation_dir_);
|
|
if (!lines_.empty())
|
|
root_handler_.ProcessAttributeUnsigned(dwarf2reader::DW_AT_stmt_list,
|
|
dwarf2reader::DW_FORM_ref4,
|
|
0);
|
|
if (language_ != dwarf2reader::DW_LANG_none) {
|
|
if (language_signed_)
|
|
root_handler_.ProcessAttributeSigned(dwarf2reader::DW_AT_language,
|
|
dwarf2reader::DW_FORM_sdata,
|
|
language_);
|
|
else
|
|
root_handler_.ProcessAttributeUnsigned(dwarf2reader::DW_AT_language,
|
|
dwarf2reader::DW_FORM_udata,
|
|
language_);
|
|
}
|
|
ASSERT_TRUE(root_handler_.EndAttributes());
|
|
}
|
|
|
|
void CUFixtureBase::ProcessStrangeAttributes(
|
|
dwarf2reader::DIEHandler *handler) {
|
|
handler->ProcessAttributeUnsigned((DwarfAttribute) 0xf560dead,
|
|
(DwarfForm) 0x4106e4db,
|
|
0xa592571997facda1ULL);
|
|
handler->ProcessAttributeSigned((DwarfAttribute) 0x85380095,
|
|
(DwarfForm) 0x0f16fe87,
|
|
0x12602a4e3bf1f446LL);
|
|
handler->ProcessAttributeReference((DwarfAttribute) 0xf7f7480f,
|
|
(DwarfForm) 0x829e038a,
|
|
0x50fddef44734fdecULL);
|
|
static const uint8_t buffer[10] = "frobynode";
|
|
handler->ProcessAttributeBuffer((DwarfAttribute) 0xa55ffb51,
|
|
(DwarfForm) 0x2f43b041,
|
|
buffer, sizeof(buffer));
|
|
handler->ProcessAttributeString((DwarfAttribute) 0x2f43b041,
|
|
(DwarfForm) 0x895ffa23,
|
|
"strange string");
|
|
}
|
|
|
|
DIEHandler *CUFixtureBase::StartNamedDIE(DIEHandler *parent,
|
|
DwarfTag tag,
|
|
const string &name) {
|
|
dwarf2reader::DIEHandler *handler
|
|
= parent->FindChildHandler(0x8f4c783c0467c989ULL, tag);
|
|
if (!handler)
|
|
return NULL;
|
|
handler->ProcessAttributeString(dwarf2reader::DW_AT_name,
|
|
dwarf2reader::DW_FORM_strp,
|
|
name);
|
|
ProcessStrangeAttributes(handler);
|
|
if (!handler->EndAttributes()) {
|
|
handler->Finish();
|
|
delete handler;
|
|
return NULL;
|
|
}
|
|
|
|
return handler;
|
|
}
|
|
|
|
DIEHandler *CUFixtureBase::StartSpecifiedDIE(DIEHandler *parent,
|
|
DwarfTag tag,
|
|
uint64 specification,
|
|
const char *name) {
|
|
dwarf2reader::DIEHandler *handler
|
|
= parent->FindChildHandler(0x8f4c783c0467c989ULL, tag);
|
|
if (!handler)
|
|
return NULL;
|
|
if (name)
|
|
handler->ProcessAttributeString(dwarf2reader::DW_AT_name,
|
|
dwarf2reader::DW_FORM_strp,
|
|
name);
|
|
handler->ProcessAttributeReference(dwarf2reader::DW_AT_specification,
|
|
dwarf2reader::DW_FORM_ref4,
|
|
specification);
|
|
if (!handler->EndAttributes()) {
|
|
handler->Finish();
|
|
delete handler;
|
|
return NULL;
|
|
}
|
|
|
|
return handler;
|
|
}
|
|
|
|
void CUFixtureBase::DefineFunction(dwarf2reader::DIEHandler *parent,
|
|
const string &name, Module::Address address,
|
|
Module::Address size,
|
|
const char* mangled_name,
|
|
DwarfForm high_pc_form) {
|
|
dwarf2reader::DIEHandler *func
|
|
= parent->FindChildHandler(0xe34797c7e68590a8LL,
|
|
dwarf2reader::DW_TAG_subprogram);
|
|
ASSERT_TRUE(func != NULL);
|
|
func->ProcessAttributeString(dwarf2reader::DW_AT_name,
|
|
dwarf2reader::DW_FORM_strp,
|
|
name);
|
|
func->ProcessAttributeUnsigned(dwarf2reader::DW_AT_low_pc,
|
|
dwarf2reader::DW_FORM_addr,
|
|
address);
|
|
|
|
Module::Address high_pc = size;
|
|
if (high_pc_form == dwarf2reader::DW_FORM_addr) {
|
|
high_pc += address;
|
|
}
|
|
func->ProcessAttributeUnsigned(dwarf2reader::DW_AT_high_pc,
|
|
high_pc_form,
|
|
high_pc);
|
|
|
|
if (mangled_name)
|
|
func->ProcessAttributeString(dwarf2reader::DW_AT_MIPS_linkage_name,
|
|
dwarf2reader::DW_FORM_strp,
|
|
mangled_name);
|
|
|
|
ProcessStrangeAttributes(func);
|
|
EXPECT_TRUE(func->EndAttributes());
|
|
func->Finish();
|
|
delete func;
|
|
}
|
|
|
|
void CUFixtureBase::DeclarationDIE(DIEHandler *parent, uint64 offset,
|
|
DwarfTag tag,
|
|
const string &name,
|
|
const string &mangled_name) {
|
|
dwarf2reader::DIEHandler *die = parent->FindChildHandler(offset, tag);
|
|
ASSERT_TRUE(die != NULL);
|
|
if (!name.empty())
|
|
die->ProcessAttributeString(dwarf2reader::DW_AT_name,
|
|
dwarf2reader::DW_FORM_strp,
|
|
name);
|
|
if (!mangled_name.empty())
|
|
die->ProcessAttributeString(dwarf2reader::DW_AT_MIPS_linkage_name,
|
|
dwarf2reader::DW_FORM_strp,
|
|
mangled_name);
|
|
|
|
die->ProcessAttributeUnsigned(dwarf2reader::DW_AT_declaration,
|
|
dwarf2reader::DW_FORM_flag,
|
|
1);
|
|
EXPECT_TRUE(die->EndAttributes());
|
|
die->Finish();
|
|
delete die;
|
|
}
|
|
|
|
void CUFixtureBase::DefinitionDIE(DIEHandler *parent,
|
|
DwarfTag tag,
|
|
uint64 specification,
|
|
const string &name,
|
|
Module::Address address,
|
|
Module::Address size) {
|
|
dwarf2reader::DIEHandler *die
|
|
= parent->FindChildHandler(0x6ccfea031a9e6cc9ULL, tag);
|
|
ASSERT_TRUE(die != NULL);
|
|
die->ProcessAttributeReference(dwarf2reader::DW_AT_specification,
|
|
dwarf2reader::DW_FORM_ref4,
|
|
specification);
|
|
if (!name.empty())
|
|
die->ProcessAttributeString(dwarf2reader::DW_AT_name,
|
|
dwarf2reader::DW_FORM_strp,
|
|
name);
|
|
if (size) {
|
|
die->ProcessAttributeUnsigned(dwarf2reader::DW_AT_low_pc,
|
|
dwarf2reader::DW_FORM_addr,
|
|
address);
|
|
die->ProcessAttributeUnsigned(dwarf2reader::DW_AT_high_pc,
|
|
dwarf2reader::DW_FORM_addr,
|
|
address + size);
|
|
}
|
|
EXPECT_TRUE(die->EndAttributes());
|
|
die->Finish();
|
|
delete die;
|
|
}
|
|
|
|
void CUFixtureBase::AbstractInstanceDIE(DIEHandler *parent,
|
|
uint64 offset,
|
|
DwarfInline type,
|
|
uint64 specification,
|
|
const string &name,
|
|
DwarfForm form) {
|
|
dwarf2reader::DIEHandler *die
|
|
= parent->FindChildHandler(offset, dwarf2reader::DW_TAG_subprogram);
|
|
ASSERT_TRUE(die != NULL);
|
|
if (specification != 0ULL)
|
|
die->ProcessAttributeReference(dwarf2reader::DW_AT_specification,
|
|
dwarf2reader::DW_FORM_ref4,
|
|
specification);
|
|
if (form == dwarf2reader::DW_FORM_sdata) {
|
|
die->ProcessAttributeSigned(dwarf2reader::DW_AT_inline, form, type);
|
|
} else {
|
|
die->ProcessAttributeUnsigned(dwarf2reader::DW_AT_inline, form, type);
|
|
}
|
|
if (!name.empty())
|
|
die->ProcessAttributeString(dwarf2reader::DW_AT_name,
|
|
dwarf2reader::DW_FORM_strp,
|
|
name);
|
|
|
|
EXPECT_TRUE(die->EndAttributes());
|
|
die->Finish();
|
|
delete die;
|
|
}
|
|
|
|
void CUFixtureBase::DefineInlineInstanceDIE(DIEHandler *parent,
|
|
const string &name,
|
|
uint64 origin,
|
|
Module::Address address,
|
|
Module::Address size) {
|
|
dwarf2reader::DIEHandler *func
|
|
= parent->FindChildHandler(0x11c70f94c6e87ccdLL,
|
|
dwarf2reader::DW_TAG_subprogram);
|
|
ASSERT_TRUE(func != NULL);
|
|
if (!name.empty()) {
|
|
func->ProcessAttributeString(dwarf2reader::DW_AT_name,
|
|
dwarf2reader::DW_FORM_strp,
|
|
name);
|
|
}
|
|
func->ProcessAttributeUnsigned(dwarf2reader::DW_AT_low_pc,
|
|
dwarf2reader::DW_FORM_addr,
|
|
address);
|
|
func->ProcessAttributeUnsigned(dwarf2reader::DW_AT_high_pc,
|
|
dwarf2reader::DW_FORM_addr,
|
|
address + size);
|
|
func->ProcessAttributeReference(dwarf2reader::DW_AT_abstract_origin,
|
|
dwarf2reader::DW_FORM_ref4,
|
|
origin);
|
|
ProcessStrangeAttributes(func);
|
|
EXPECT_TRUE(func->EndAttributes());
|
|
func->Finish();
|
|
delete func;
|
|
}
|
|
|
|
void CUFixtureBase::FillFunctions() {
|
|
if (functions_filled_)
|
|
return;
|
|
module_.GetFunctions(&functions_, functions_.end());
|
|
sort(functions_.begin(), functions_.end(),
|
|
Module::Function::CompareByAddress);
|
|
functions_filled_ = true;
|
|
}
|
|
|
|
void CUFixtureBase::TestFunctionCount(size_t expected) {
|
|
FillFunctions();
|
|
ASSERT_EQ(expected, functions_.size());
|
|
}
|
|
|
|
void CUFixtureBase::TestFunction(int i, const string &name,
|
|
Module::Address address,
|
|
Module::Address size) {
|
|
FillFunctions();
|
|
ASSERT_LT((size_t) i, functions_.size());
|
|
|
|
Module::Function *function = functions_[i];
|
|
EXPECT_EQ(name, function->name);
|
|
EXPECT_EQ(address, function->address);
|
|
EXPECT_EQ(size, function->size);
|
|
EXPECT_EQ(0U, function->parameter_size);
|
|
}
|
|
|
|
void CUFixtureBase::TestLineCount(int i, size_t expected) {
|
|
FillFunctions();
|
|
ASSERT_LT((size_t) i, functions_.size());
|
|
|
|
ASSERT_EQ(expected, functions_[i]->lines.size());
|
|
}
|
|
|
|
void CUFixtureBase::TestLine(int i, int j,
|
|
Module::Address address, Module::Address size,
|
|
const string &filename, int number) {
|
|
FillFunctions();
|
|
ASSERT_LT((size_t) i, functions_.size());
|
|
ASSERT_LT((size_t) j, functions_[i]->lines.size());
|
|
|
|
Module::Line *line = &functions_[i]->lines[j];
|
|
EXPECT_EQ(address, line->address);
|
|
EXPECT_EQ(size, line->size);
|
|
EXPECT_EQ(filename, line->file->name.c_str());
|
|
EXPECT_EQ(number, line->number);
|
|
}
|
|
|
|
// Include caller locations for our test subroutines.
|
|
#define TRACE(call) do { SCOPED_TRACE("called from here"); call; } while (0)
|
|
#define PushLine(a,b,c,d) TRACE(PushLine((a),(b),(c),(d)))
|
|
#define SetLanguage(a) TRACE(SetLanguage(a))
|
|
#define StartCU() TRACE(StartCU())
|
|
#define DefineFunction(a,b,c,d,e) TRACE(DefineFunction((a),(b),(c),(d),(e)))
|
|
// (DefineFunction) instead of DefineFunction to avoid macro expansion.
|
|
#define DefineFunction6(a,b,c,d,e,f) \
|
|
TRACE((DefineFunction)((a),(b),(c),(d),(e),(f)))
|
|
#define DeclarationDIE(a,b,c,d,e) TRACE(DeclarationDIE((a),(b),(c),(d),(e)))
|
|
#define DefinitionDIE(a,b,c,d,e,f) \
|
|
TRACE(DefinitionDIE((a),(b),(c),(d),(e),(f)))
|
|
#define TestFunctionCount(a) TRACE(TestFunctionCount(a))
|
|
#define TestFunction(a,b,c,d) TRACE(TestFunction((a),(b),(c),(d)))
|
|
#define TestLineCount(a,b) TRACE(TestLineCount((a),(b)))
|
|
#define TestLine(a,b,c,d,e,f) TRACE(TestLine((a),(b),(c),(d),(e),(f)))
|
|
|
|
class SimpleCU: public CUFixtureBase, public Test {
|
|
};
|
|
|
|
TEST_F(SimpleCU, CompilationDir) {
|
|
compilation_dir_ = "/src/build/";
|
|
|
|
StartCU();
|
|
root_handler_.Finish();
|
|
}
|
|
|
|
TEST_F(SimpleCU, OneFunc) {
|
|
PushLine(0x938cf8c07def4d34ULL, 0x55592d727f6cd01fLL, "line-file", 246571772);
|
|
|
|
StartCU();
|
|
DefineFunction(&root_handler_, "function1",
|
|
0x938cf8c07def4d34ULL, 0x55592d727f6cd01fLL, NULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "function1", 0x938cf8c07def4d34ULL, 0x55592d727f6cd01fLL);
|
|
TestLineCount(0, 1);
|
|
TestLine(0, 0, 0x938cf8c07def4d34ULL, 0x55592d727f6cd01fLL, "line-file",
|
|
246571772);
|
|
}
|
|
|
|
// As above, only DW_AT_high_pc is a length rather than an address.
|
|
TEST_F(SimpleCU, OneFuncHighPcIsLength) {
|
|
PushLine(0x938cf8c07def4d34ULL, 0x55592d727f6cd01fLL, "line-file", 246571772);
|
|
|
|
StartCU();
|
|
DefineFunction6(&root_handler_, "function1",
|
|
0x938cf8c07def4d34ULL, 0x55592d727f6cd01fLL, NULL,
|
|
dwarf2reader::DW_FORM_udata);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "function1", 0x938cf8c07def4d34ULL, 0x55592d727f6cd01fLL);
|
|
TestLineCount(0, 1);
|
|
TestLine(0, 0, 0x938cf8c07def4d34ULL, 0x55592d727f6cd01fLL, "line-file",
|
|
246571772);
|
|
}
|
|
|
|
TEST_F(SimpleCU, MangledName) {
|
|
PushLine(0x938cf8c07def4d34ULL, 0x55592d727f6cd01fLL, "line-file", 246571772);
|
|
|
|
StartCU();
|
|
DefineFunction(&root_handler_, "function1",
|
|
0x938cf8c07def4d34ULL, 0x55592d727f6cd01fLL, "_ZN1n1fEi");
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "n::f(int)", 0x938cf8c07def4d34ULL, 0x55592d727f6cd01fLL);
|
|
}
|
|
|
|
TEST_F(SimpleCU, IrrelevantRootChildren) {
|
|
StartCU();
|
|
EXPECT_FALSE(root_handler_
|
|
.FindChildHandler(0x7db32bff4e2dcfb1ULL,
|
|
dwarf2reader::DW_TAG_lexical_block));
|
|
}
|
|
|
|
TEST_F(SimpleCU, IrrelevantNamedScopeChildren) {
|
|
StartCU();
|
|
DIEHandler *class_A_handler
|
|
= StartNamedDIE(&root_handler_, dwarf2reader::DW_TAG_class_type, "class_A");
|
|
EXPECT_TRUE(class_A_handler != NULL);
|
|
EXPECT_FALSE(class_A_handler
|
|
->FindChildHandler(0x02e55999b865e4e9ULL,
|
|
dwarf2reader::DW_TAG_lexical_block));
|
|
delete class_A_handler;
|
|
}
|
|
|
|
// Verify that FileContexts can safely be deleted unused.
|
|
TEST_F(SimpleCU, UnusedFileContext) {
|
|
Module m("module-name", "module-os", "module-arch", "module-id");
|
|
DwarfCUToModule::FileContext fc("dwarf-filename", &m, true);
|
|
|
|
// Kludge: satisfy reporter_'s expectation.
|
|
reporter_.SetCUName("compilation-unit-name");
|
|
}
|
|
|
|
TEST_F(SimpleCU, InlineFunction) {
|
|
PushLine(0x1758a0f941b71efbULL, 0x1cf154f1f545e146ULL, "line-file", 75173118);
|
|
|
|
StartCU();
|
|
AbstractInstanceDIE(&root_handler_, 0x1e8dac5d507ed7abULL,
|
|
dwarf2reader::DW_INL_inlined, 0, "inline-name");
|
|
DefineInlineInstanceDIE(&root_handler_, "", 0x1e8dac5d507ed7abULL,
|
|
0x1758a0f941b71efbULL, 0x1cf154f1f545e146ULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "inline-name",
|
|
0x1758a0f941b71efbULL, 0x1cf154f1f545e146ULL);
|
|
}
|
|
|
|
TEST_F(SimpleCU, InlineFunctionSignedAttribute) {
|
|
PushLine(0x1758a0f941b71efbULL, 0x1cf154f1f545e146ULL, "line-file", 75173118);
|
|
|
|
StartCU();
|
|
AbstractInstanceDIE(&root_handler_, 0x1e8dac5d507ed7abULL,
|
|
dwarf2reader::DW_INL_inlined, 0, "inline-name",
|
|
dwarf2reader::DW_FORM_sdata);
|
|
DefineInlineInstanceDIE(&root_handler_, "", 0x1e8dac5d507ed7abULL,
|
|
0x1758a0f941b71efbULL, 0x1cf154f1f545e146ULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "inline-name",
|
|
0x1758a0f941b71efbULL, 0x1cf154f1f545e146ULL);
|
|
}
|
|
|
|
// Any DIE with an DW_AT_inline attribute can be cited by
|
|
// DW_AT_abstract_origin attributes --- even if the value of the
|
|
// DW_AT_inline attribute is DW_INL_not_inlined.
|
|
TEST_F(SimpleCU, AbstractOriginNotInlined) {
|
|
PushLine(0x2805c4531be6ca0eULL, 0x686b52155a8d4d2cULL, "line-file", 6111581);
|
|
|
|
StartCU();
|
|
AbstractInstanceDIE(&root_handler_, 0x93e9cdad52826b39ULL,
|
|
dwarf2reader::DW_INL_not_inlined, 0, "abstract-instance");
|
|
DefineInlineInstanceDIE(&root_handler_, "", 0x93e9cdad52826b39ULL,
|
|
0x2805c4531be6ca0eULL, 0x686b52155a8d4d2cULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "abstract-instance",
|
|
0x2805c4531be6ca0eULL, 0x686b52155a8d4d2cULL);
|
|
}
|
|
|
|
TEST_F(SimpleCU, UnknownAbstractOrigin) {
|
|
EXPECT_CALL(reporter_, UnknownAbstractOrigin(_, 1ULL)).WillOnce(Return());
|
|
EXPECT_CALL(reporter_, UnnamedFunction(0x11c70f94c6e87ccdLL))
|
|
.WillOnce(Return());
|
|
PushLine(0x1758a0f941b71efbULL, 0x1cf154f1f545e146ULL, "line-file", 75173118);
|
|
|
|
StartCU();
|
|
AbstractInstanceDIE(&root_handler_, 0x1e8dac5d507ed7abULL,
|
|
dwarf2reader::DW_INL_inlined, 0, "inline-name");
|
|
DefineInlineInstanceDIE(&root_handler_, "", 1ULL,
|
|
0x1758a0f941b71efbULL, 0x1cf154f1f545e146ULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "<name omitted>",
|
|
0x1758a0f941b71efbULL, 0x1cf154f1f545e146ULL);
|
|
}
|
|
|
|
TEST_F(SimpleCU, UnnamedFunction) {
|
|
EXPECT_CALL(reporter_, UnnamedFunction(0xe34797c7e68590a8LL))
|
|
.WillOnce(Return());
|
|
PushLine(0x72b80e41a0ac1d40ULL, 0x537174f231ee181cULL, "line-file", 14044850);
|
|
|
|
StartCU();
|
|
DefineFunction(&root_handler_, "",
|
|
0x72b80e41a0ac1d40ULL, 0x537174f231ee181cULL, NULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "<name omitted>",
|
|
0x72b80e41a0ac1d40ULL, 0x537174f231ee181cULL);
|
|
}
|
|
|
|
// An address range.
|
|
struct Range {
|
|
Module::Address start, end;
|
|
};
|
|
|
|
// Test data for pairing functions and lines.
|
|
struct Situation {
|
|
// Two function intervals, and two line intervals.
|
|
Range functions[2], lines[2];
|
|
|
|
// The number of lines we expect to be assigned to each of the
|
|
// functions, and the address ranges.
|
|
int paired_count[2];
|
|
Range paired[2][2];
|
|
|
|
// The number of functions that are not entirely covered by lines,
|
|
// and vice versa.
|
|
int uncovered_functions, uncovered_lines;
|
|
};
|
|
|
|
#define PAIRING(func1_start, func1_end, func2_start, func2_end, \
|
|
line1_start, line1_end, line2_start, line2_end, \
|
|
func1_num_lines, func2_num_lines, \
|
|
func1_line1_start, func1_line1_end, \
|
|
func1_line2_start, func1_line2_end, \
|
|
func2_line1_start, func2_line1_end, \
|
|
func2_line2_start, func2_line2_end, \
|
|
uncovered_functions, uncovered_lines) \
|
|
{ { { func1_start, func1_end }, { func2_start, func2_end } }, \
|
|
{ { line1_start, line1_end }, { line2_start, line2_end } }, \
|
|
{ func1_num_lines, func2_num_lines }, \
|
|
{ { { func1_line1_start, func1_line1_end }, \
|
|
{ func1_line2_start, func1_line2_end } }, \
|
|
{ { func2_line1_start, func2_line1_end }, \
|
|
{ func2_line2_start, func2_line2_end } } }, \
|
|
uncovered_functions, uncovered_lines },
|
|
|
|
Situation situations[] = {
|
|
#include "common/testdata/func-line-pairing.h"
|
|
};
|
|
|
|
#undef PAIRING
|
|
|
|
class FuncLinePairing: public CUFixtureBase,
|
|
public TestWithParam<Situation> { };
|
|
|
|
INSTANTIATE_TEST_CASE_P(AllSituations, FuncLinePairing,
|
|
ValuesIn(situations));
|
|
|
|
TEST_P(FuncLinePairing, Pairing) {
|
|
const Situation &s = GetParam();
|
|
PushLine(s.lines[0].start,
|
|
s.lines[0].end - s.lines[0].start,
|
|
"line-file", 67636963);
|
|
PushLine(s.lines[1].start,
|
|
s.lines[1].end - s.lines[1].start,
|
|
"line-file", 67636963);
|
|
if (s.uncovered_functions)
|
|
EXPECT_CALL(reporter_, UncoveredFunction(_))
|
|
.Times(s.uncovered_functions)
|
|
.WillRepeatedly(Return());
|
|
if (s.uncovered_lines)
|
|
EXPECT_CALL(reporter_, UncoveredLine(_))
|
|
.Times(s.uncovered_lines)
|
|
.WillRepeatedly(Return());
|
|
|
|
StartCU();
|
|
DefineFunction(&root_handler_, "function1",
|
|
s.functions[0].start,
|
|
s.functions[0].end - s.functions[0].start, NULL);
|
|
DefineFunction(&root_handler_, "function2",
|
|
s.functions[1].start,
|
|
s.functions[1].end - s.functions[1].start, NULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(2);
|
|
TestFunction(0, "function1",
|
|
s.functions[0].start,
|
|
s.functions[0].end - s.functions[0].start);
|
|
TestLineCount(0, s.paired_count[0]);
|
|
for (int i = 0; i < s.paired_count[0]; i++)
|
|
TestLine(0, i, s.paired[0][i].start,
|
|
s.paired[0][i].end - s.paired[0][i].start,
|
|
"line-file", 67636963);
|
|
TestFunction(1, "function2",
|
|
s.functions[1].start,
|
|
s.functions[1].end - s.functions[1].start);
|
|
TestLineCount(1, s.paired_count[1]);
|
|
for (int i = 0; i < s.paired_count[1]; i++)
|
|
TestLine(1, i, s.paired[1][i].start,
|
|
s.paired[1][i].end - s.paired[1][i].start,
|
|
"line-file", 67636963);
|
|
}
|
|
|
|
TEST_F(FuncLinePairing, EmptyCU) {
|
|
StartCU();
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(0);
|
|
}
|
|
|
|
TEST_F(FuncLinePairing, LinesNoFuncs) {
|
|
PushLine(40, 2, "line-file", 82485646);
|
|
EXPECT_CALL(reporter_, UncoveredLine(_)).WillOnce(Return());
|
|
|
|
StartCU();
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(0);
|
|
}
|
|
|
|
TEST_F(FuncLinePairing, FuncsNoLines) {
|
|
EXPECT_CALL(reporter_, UncoveredFunction(_)).WillOnce(Return());
|
|
|
|
StartCU();
|
|
DefineFunction(&root_handler_, "function1", 0x127da12ffcf5c51fULL, 0x1000U,
|
|
NULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "function1", 0x127da12ffcf5c51fULL, 0x1000U);
|
|
}
|
|
|
|
TEST_F(FuncLinePairing, GapThenFunction) {
|
|
PushLine(20, 2, "line-file-2", 174314698);
|
|
PushLine(10, 2, "line-file-1", 263008005);
|
|
|
|
StartCU();
|
|
DefineFunction(&root_handler_, "function1", 10, 2, NULL);
|
|
DefineFunction(&root_handler_, "function2", 20, 2, NULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(2);
|
|
TestFunction(0, "function1", 10, 2);
|
|
TestLineCount(0, 1);
|
|
TestLine(0, 0, 10, 2, "line-file-1", 263008005);
|
|
TestFunction(1, "function2", 20, 2);
|
|
TestLineCount(1, 1);
|
|
TestLine(1, 0, 20, 2, "line-file-2", 174314698);
|
|
}
|
|
|
|
// If GCC emits padding after one function to align the start of
|
|
// the next, then it will attribute the padding instructions to
|
|
// the last source line of function (to reduce the size of the
|
|
// line number info), but omit it from the DW_AT_{low,high}_pc
|
|
// range given in .debug_info (since it costs nothing to be
|
|
// precise there). If we did use at least some of the line
|
|
// we're about to skip, then assume this is what happened, and
|
|
// don't warn.
|
|
TEST_F(FuncLinePairing, GCCAlignmentStretch) {
|
|
PushLine(10, 10, "line-file", 63351048);
|
|
PushLine(20, 10, "line-file", 61661044);
|
|
|
|
StartCU();
|
|
DefineFunction(&root_handler_, "function1", 10, 5, NULL);
|
|
// five-byte gap between functions, covered by line 63351048.
|
|
// This should not elicit a warning.
|
|
DefineFunction(&root_handler_, "function2", 20, 10, NULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(2);
|
|
TestFunction(0, "function1", 10, 5);
|
|
TestLineCount(0, 1);
|
|
TestLine(0, 0, 10, 5, "line-file", 63351048);
|
|
TestFunction(1, "function2", 20, 10);
|
|
TestLineCount(1, 1);
|
|
TestLine(1, 0, 20, 10, "line-file", 61661044);
|
|
}
|
|
|
|
// Unfortunately, neither the DWARF parser's handler interface nor the
|
|
// DIEHandler interface is capable of expressing a function that abuts
|
|
// the end of the address space: the high_pc value looks like zero.
|
|
|
|
TEST_F(FuncLinePairing, LineAtEndOfAddressSpace) {
|
|
PushLine(0xfffffffffffffff0ULL, 16, "line-file", 63351048);
|
|
EXPECT_CALL(reporter_, UncoveredLine(_)).WillOnce(Return());
|
|
|
|
StartCU();
|
|
DefineFunction(&root_handler_, "function1", 0xfffffffffffffff0ULL, 6, NULL);
|
|
DefineFunction(&root_handler_, "function2", 0xfffffffffffffffaULL, 5, NULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(2);
|
|
TestFunction(0, "function1", 0xfffffffffffffff0ULL, 6);
|
|
TestLineCount(0, 1);
|
|
TestLine(0, 0, 0xfffffffffffffff0ULL, 6, "line-file", 63351048);
|
|
TestFunction(1, "function2", 0xfffffffffffffffaULL, 5);
|
|
TestLineCount(1, 1);
|
|
TestLine(1, 0, 0xfffffffffffffffaULL, 5, "line-file", 63351048);
|
|
}
|
|
|
|
// A function with more than one uncovered area should only be warned
|
|
// about once.
|
|
TEST_F(FuncLinePairing, WarnOnceFunc) {
|
|
PushLine(20, 1, "line-file-2", 262951329);
|
|
PushLine(11, 1, "line-file-1", 219964021);
|
|
EXPECT_CALL(reporter_, UncoveredFunction(_)).WillOnce(Return());
|
|
|
|
StartCU();
|
|
DefineFunction(&root_handler_, "function", 10, 11, NULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "function", 10, 11);
|
|
TestLineCount(0, 2);
|
|
TestLine(0, 0, 11, 1, "line-file-1", 219964021);
|
|
TestLine(0, 1, 20, 1, "line-file-2", 262951329);
|
|
}
|
|
|
|
// A line with more than one uncovered area should only be warned
|
|
// about once.
|
|
TEST_F(FuncLinePairing, WarnOnceLine) {
|
|
PushLine(10, 20, "filename1", 118581871);
|
|
EXPECT_CALL(reporter_, UncoveredLine(_)).WillOnce(Return());
|
|
|
|
StartCU();
|
|
DefineFunction(&root_handler_, "function1", 11, 1, NULL);
|
|
DefineFunction(&root_handler_, "function2", 13, 1, NULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(2);
|
|
TestFunction(0, "function1", 11, 1);
|
|
TestLineCount(0, 1);
|
|
TestLine(0, 0, 11, 1, "filename1", 118581871);
|
|
TestFunction(1, "function2", 13, 1);
|
|
TestLineCount(1, 1);
|
|
TestLine(1, 0, 13, 1, "filename1", 118581871);
|
|
}
|
|
|
|
class CXXQualifiedNames: public CUFixtureBase,
|
|
public TestWithParam<DwarfTag> { };
|
|
|
|
INSTANTIATE_TEST_CASE_P(VersusEnclosures, CXXQualifiedNames,
|
|
Values(dwarf2reader::DW_TAG_class_type,
|
|
dwarf2reader::DW_TAG_structure_type,
|
|
dwarf2reader::DW_TAG_union_type,
|
|
dwarf2reader::DW_TAG_namespace));
|
|
|
|
TEST_P(CXXQualifiedNames, TwoFunctions) {
|
|
DwarfTag tag = GetParam();
|
|
|
|
SetLanguage(dwarf2reader::DW_LANG_C_plus_plus);
|
|
PushLine(10, 1, "filename1", 69819327);
|
|
PushLine(20, 1, "filename2", 95115701);
|
|
|
|
StartCU();
|
|
DIEHandler *enclosure_handler = StartNamedDIE(&root_handler_, tag,
|
|
"Enclosure");
|
|
EXPECT_TRUE(enclosure_handler != NULL);
|
|
DefineFunction(enclosure_handler, "func_B", 10, 1, NULL);
|
|
DefineFunction(enclosure_handler, "func_C", 20, 1, NULL);
|
|
enclosure_handler->Finish();
|
|
delete enclosure_handler;
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(2);
|
|
TestFunction(0, "Enclosure::func_B", 10, 1);
|
|
TestFunction(1, "Enclosure::func_C", 20, 1);
|
|
}
|
|
|
|
TEST_P(CXXQualifiedNames, FuncInEnclosureInNamespace) {
|
|
DwarfTag tag = GetParam();
|
|
|
|
SetLanguage(dwarf2reader::DW_LANG_C_plus_plus);
|
|
PushLine(10, 1, "line-file", 69819327);
|
|
|
|
StartCU();
|
|
DIEHandler *namespace_handler
|
|
= StartNamedDIE(&root_handler_, dwarf2reader::DW_TAG_namespace,
|
|
"Namespace");
|
|
EXPECT_TRUE(namespace_handler != NULL);
|
|
DIEHandler *enclosure_handler = StartNamedDIE(namespace_handler, tag,
|
|
"Enclosure");
|
|
EXPECT_TRUE(enclosure_handler != NULL);
|
|
DefineFunction(enclosure_handler, "function", 10, 1, NULL);
|
|
enclosure_handler->Finish();
|
|
delete enclosure_handler;
|
|
namespace_handler->Finish();
|
|
delete namespace_handler;
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "Namespace::Enclosure::function", 10, 1);
|
|
}
|
|
|
|
TEST_F(CXXQualifiedNames, FunctionInClassInStructInNamespace) {
|
|
SetLanguage(dwarf2reader::DW_LANG_C_plus_plus);
|
|
PushLine(10, 1, "filename1", 69819327);
|
|
|
|
StartCU();
|
|
DIEHandler *namespace_handler
|
|
= StartNamedDIE(&root_handler_, dwarf2reader::DW_TAG_namespace,
|
|
"namespace_A");
|
|
EXPECT_TRUE(namespace_handler != NULL);
|
|
DIEHandler *struct_handler
|
|
= StartNamedDIE(namespace_handler, dwarf2reader::DW_TAG_structure_type,
|
|
"struct_B");
|
|
EXPECT_TRUE(struct_handler != NULL);
|
|
DIEHandler *class_handler
|
|
= StartNamedDIE(struct_handler, dwarf2reader::DW_TAG_class_type,
|
|
"class_C");
|
|
DefineFunction(class_handler, "function_D", 10, 1, NULL);
|
|
class_handler->Finish();
|
|
delete class_handler;
|
|
struct_handler->Finish();
|
|
delete struct_handler;
|
|
namespace_handler->Finish();
|
|
delete namespace_handler;
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "namespace_A::struct_B::class_C::function_D", 10, 1);
|
|
}
|
|
|
|
struct LanguageAndQualifiedName {
|
|
dwarf2reader::DwarfLanguage language;
|
|
const char *name;
|
|
};
|
|
|
|
const LanguageAndQualifiedName LanguageAndQualifiedNameCases[] = {
|
|
{ dwarf2reader::DW_LANG_none, "class_A::function_B" },
|
|
{ dwarf2reader::DW_LANG_C, "class_A::function_B" },
|
|
{ dwarf2reader::DW_LANG_C89, "class_A::function_B" },
|
|
{ dwarf2reader::DW_LANG_C99, "class_A::function_B" },
|
|
{ dwarf2reader::DW_LANG_C_plus_plus, "class_A::function_B" },
|
|
{ dwarf2reader::DW_LANG_Java, "class_A.function_B" },
|
|
{ dwarf2reader::DW_LANG_Cobol74, "class_A::function_B" },
|
|
{ dwarf2reader::DW_LANG_Mips_Assembler, NULL }
|
|
};
|
|
|
|
class QualifiedForLanguage
|
|
: public CUFixtureBase,
|
|
public TestWithParam<LanguageAndQualifiedName> { };
|
|
|
|
INSTANTIATE_TEST_CASE_P(LanguageAndQualifiedName, QualifiedForLanguage,
|
|
ValuesIn(LanguageAndQualifiedNameCases));
|
|
|
|
TEST_P(QualifiedForLanguage, MemberFunction) {
|
|
const LanguageAndQualifiedName ¶m = GetParam();
|
|
|
|
PushLine(10, 1, "line-file", 212966758);
|
|
SetLanguage(param.language);
|
|
|
|
StartCU();
|
|
DIEHandler *class_handler
|
|
= StartNamedDIE(&root_handler_, dwarf2reader::DW_TAG_class_type,
|
|
"class_A");
|
|
DefineFunction(class_handler, "function_B", 10, 1, NULL);
|
|
class_handler->Finish();
|
|
delete class_handler;
|
|
root_handler_.Finish();
|
|
|
|
if (param.name) {
|
|
TestFunctionCount(1);
|
|
TestFunction(0, param.name, 10, 1);
|
|
} else {
|
|
TestFunctionCount(0);
|
|
}
|
|
}
|
|
|
|
TEST_P(QualifiedForLanguage, MemberFunctionSignedLanguage) {
|
|
const LanguageAndQualifiedName ¶m = GetParam();
|
|
|
|
PushLine(10, 1, "line-file", 212966758);
|
|
SetLanguage(param.language);
|
|
SetLanguageSigned(true);
|
|
|
|
StartCU();
|
|
DIEHandler *class_handler
|
|
= StartNamedDIE(&root_handler_, dwarf2reader::DW_TAG_class_type,
|
|
"class_A");
|
|
DefineFunction(class_handler, "function_B", 10, 1, NULL);
|
|
class_handler->Finish();
|
|
delete class_handler;
|
|
root_handler_.Finish();
|
|
|
|
if (param.name) {
|
|
TestFunctionCount(1);
|
|
TestFunction(0, param.name, 10, 1);
|
|
} else {
|
|
TestFunctionCount(0);
|
|
}
|
|
}
|
|
|
|
class Specifications: public CUFixtureBase, public Test { };
|
|
|
|
TEST_F(Specifications, Function) {
|
|
PushLine(0x93cd3dfc1aa10097ULL, 0x0397d47a0b4ca0d4ULL, "line-file", 54883661);
|
|
|
|
StartCU();
|
|
DeclarationDIE(&root_handler_, 0xcd3c51b946fb1eeeLL,
|
|
dwarf2reader::DW_TAG_subprogram, "declaration-name", "");
|
|
DefinitionDIE(&root_handler_, dwarf2reader::DW_TAG_subprogram,
|
|
0xcd3c51b946fb1eeeLL, "",
|
|
0x93cd3dfc1aa10097ULL, 0x0397d47a0b4ca0d4ULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "declaration-name",
|
|
0x93cd3dfc1aa10097ULL, 0x0397d47a0b4ca0d4ULL);
|
|
}
|
|
|
|
TEST_F(Specifications, MangledName) {
|
|
// Language defaults to C++, so no need to set it here.
|
|
PushLine(0x93cd3dfc1aa10097ULL, 0x0397d47a0b4ca0d4ULL, "line-file", 54883661);
|
|
|
|
StartCU();
|
|
DeclarationDIE(&root_handler_, 0xcd3c51b946fb1eeeLL,
|
|
dwarf2reader::DW_TAG_subprogram, "declaration-name",
|
|
"_ZN1C1fEi");
|
|
DefinitionDIE(&root_handler_, dwarf2reader::DW_TAG_subprogram,
|
|
0xcd3c51b946fb1eeeLL, "",
|
|
0x93cd3dfc1aa10097ULL, 0x0397d47a0b4ca0d4ULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "C::f(int)",
|
|
0x93cd3dfc1aa10097ULL, 0x0397d47a0b4ca0d4ULL);
|
|
}
|
|
|
|
TEST_F(Specifications, MangledNameSwift) {
|
|
// Swift mangled names should pass through untouched.
|
|
SetLanguage(dwarf2reader::DW_LANG_Swift);
|
|
PushLine(0x93cd3dfc1aa10097ULL, 0x0397d47a0b4ca0d4ULL, "line-file", 54883661);
|
|
StartCU();
|
|
const string kName = "_TFC9swifttest5Shape17simpleDescriptionfS0_FT_Si";
|
|
DeclarationDIE(&root_handler_, 0xcd3c51b946fb1eeeLL,
|
|
dwarf2reader::DW_TAG_subprogram, "declaration-name",
|
|
kName);
|
|
DefinitionDIE(&root_handler_, dwarf2reader::DW_TAG_subprogram,
|
|
0xcd3c51b946fb1eeeLL, "",
|
|
0x93cd3dfc1aa10097ULL, 0x0397d47a0b4ca0d4ULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, kName,
|
|
0x93cd3dfc1aa10097ULL, 0x0397d47a0b4ca0d4ULL);
|
|
}
|
|
|
|
TEST_F(Specifications, MangledNameRust) {
|
|
SetLanguage(dwarf2reader::DW_LANG_Rust);
|
|
PushLine(0x93cd3dfc1aa10097ULL, 0x0397d47a0b4ca0d4ULL, "line-file", 54883661);
|
|
|
|
StartCU();
|
|
const string kName = "_ZN14rustc_demangle8demangle17h373defa94bffacdeE";
|
|
DeclarationDIE(&root_handler_, 0xcd3c51b946fb1eeeLL,
|
|
dwarf2reader::DW_TAG_subprogram, "declaration-name",
|
|
kName);
|
|
DefinitionDIE(&root_handler_, dwarf2reader::DW_TAG_subprogram,
|
|
0xcd3c51b946fb1eeeLL, "",
|
|
0x93cd3dfc1aa10097ULL, 0x0397d47a0b4ca0d4ULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0,
|
|
#ifndef HAVE_RUST_DEMANGLE
|
|
// Rust mangled names should pass through untouched if not
|
|
// using rust-demangle.
|
|
kName,
|
|
#else
|
|
// If rust-demangle is available this should be properly
|
|
// demangled.
|
|
"rustc_demangle::demangle",
|
|
#endif
|
|
0x93cd3dfc1aa10097ULL, 0x0397d47a0b4ca0d4ULL);
|
|
}
|
|
|
|
TEST_F(Specifications, MemberFunction) {
|
|
PushLine(0x3341a248634e7170ULL, 0x5f6938ee5553b953ULL, "line-file", 18116691);
|
|
|
|
StartCU();
|
|
DIEHandler *class_handler
|
|
= StartNamedDIE(&root_handler_, dwarf2reader::DW_TAG_class_type, "class_A");
|
|
DeclarationDIE(class_handler, 0x7d83028c431406e8ULL,
|
|
dwarf2reader::DW_TAG_subprogram, "declaration-name", "");
|
|
class_handler->Finish();
|
|
delete class_handler;
|
|
DefinitionDIE(&root_handler_, dwarf2reader::DW_TAG_subprogram,
|
|
0x7d83028c431406e8ULL, "",
|
|
0x3341a248634e7170ULL, 0x5f6938ee5553b953ULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "class_A::declaration-name",
|
|
0x3341a248634e7170ULL, 0x5f6938ee5553b953ULL);
|
|
}
|
|
|
|
// This case should gather the name from both the definition and the
|
|
// declaration's parent.
|
|
TEST_F(Specifications, FunctionDeclarationParent) {
|
|
PushLine(0x463c9ddf405be227ULL, 0x6a47774af5049680ULL, "line-file", 70254922);
|
|
|
|
StartCU();
|
|
{
|
|
DIEHandler *class_handler
|
|
= StartNamedDIE(&root_handler_, dwarf2reader::DW_TAG_class_type,
|
|
"class_A");
|
|
ASSERT_TRUE(class_handler != NULL);
|
|
DeclarationDIE(class_handler, 0x0e0e877c8404544aULL,
|
|
dwarf2reader::DW_TAG_subprogram, "declaration-name", "");
|
|
class_handler->Finish();
|
|
delete class_handler;
|
|
}
|
|
|
|
DefinitionDIE(&root_handler_, dwarf2reader::DW_TAG_subprogram,
|
|
0x0e0e877c8404544aULL, "definition-name",
|
|
0x463c9ddf405be227ULL, 0x6a47774af5049680ULL);
|
|
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "class_A::definition-name",
|
|
0x463c9ddf405be227ULL, 0x6a47774af5049680ULL);
|
|
}
|
|
|
|
// Named scopes should also gather enclosing name components from
|
|
// their declarations.
|
|
TEST_F(Specifications, NamedScopeDeclarationParent) {
|
|
PushLine(0x5d13433d0df13d00ULL, 0x48ebebe5ade2cab4ULL, "line-file", 77392604);
|
|
|
|
StartCU();
|
|
{
|
|
DIEHandler *space_handler
|
|
= StartNamedDIE(&root_handler_, dwarf2reader::DW_TAG_namespace,
|
|
"space_A");
|
|
ASSERT_TRUE(space_handler != NULL);
|
|
DeclarationDIE(space_handler, 0x419bb1d12f9a73a2ULL,
|
|
dwarf2reader::DW_TAG_class_type, "class-declaration-name",
|
|
"");
|
|
space_handler->Finish();
|
|
delete space_handler;
|
|
}
|
|
|
|
{
|
|
DIEHandler *class_handler
|
|
= StartSpecifiedDIE(&root_handler_, dwarf2reader::DW_TAG_class_type,
|
|
0x419bb1d12f9a73a2ULL, "class-definition-name");
|
|
ASSERT_TRUE(class_handler != NULL);
|
|
DefineFunction(class_handler, "function",
|
|
0x5d13433d0df13d00ULL, 0x48ebebe5ade2cab4ULL, NULL);
|
|
class_handler->Finish();
|
|
delete class_handler;
|
|
}
|
|
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "space_A::class-definition-name::function",
|
|
0x5d13433d0df13d00ULL, 0x48ebebe5ade2cab4ULL);
|
|
}
|
|
|
|
// This test recreates bug 364.
|
|
TEST_F(Specifications, InlineFunction) {
|
|
PushLine(0x1758a0f941b71efbULL, 0x1cf154f1f545e146ULL, "line-file", 75173118);
|
|
|
|
StartCU();
|
|
DeclarationDIE(&root_handler_, 0xcd3c51b946fb1eeeLL,
|
|
dwarf2reader::DW_TAG_subprogram, "inline-name", "");
|
|
AbstractInstanceDIE(&root_handler_, 0x1e8dac5d507ed7abULL,
|
|
dwarf2reader::DW_INL_inlined, 0xcd3c51b946fb1eeeLL, "");
|
|
DefineInlineInstanceDIE(&root_handler_, "", 0x1e8dac5d507ed7abULL,
|
|
0x1758a0f941b71efbULL, 0x1cf154f1f545e146ULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "inline-name",
|
|
0x1758a0f941b71efbULL, 0x1cf154f1f545e146ULL);
|
|
}
|
|
|
|
// An inline function in a namespace should correctly derive its
|
|
// name from its abstract origin, and not just the namespace name.
|
|
TEST_F(Specifications, InlineFunctionInNamespace) {
|
|
PushLine(0x1758a0f941b71efbULL, 0x1cf154f1f545e146ULL, "line-file", 75173118);
|
|
|
|
StartCU();
|
|
DIEHandler* space_handler
|
|
= StartNamedDIE(&root_handler_, dwarf2reader::DW_TAG_namespace,
|
|
"Namespace");
|
|
ASSERT_TRUE(space_handler != NULL);
|
|
AbstractInstanceDIE(space_handler, 0x1e8dac5d507ed7abULL,
|
|
dwarf2reader::DW_INL_inlined, 0LL, "func-name");
|
|
DefineInlineInstanceDIE(space_handler, "", 0x1e8dac5d507ed7abULL,
|
|
0x1758a0f941b71efbULL, 0x1cf154f1f545e146ULL);
|
|
space_handler->Finish();
|
|
delete space_handler;
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "Namespace::func-name",
|
|
0x1758a0f941b71efbULL, 0x1cf154f1f545e146ULL);
|
|
}
|
|
|
|
// Check name construction for a long chain containing each combination of:
|
|
// - struct, union, class, namespace
|
|
// - direct and definition
|
|
TEST_F(Specifications, LongChain) {
|
|
PushLine(0x5a0dd6bb85db754cULL, 0x3bccb213d08c7fd3ULL, "line-file", 21192926);
|
|
SetLanguage(dwarf2reader::DW_LANG_C_plus_plus);
|
|
|
|
StartCU();
|
|
// The structure we're building here is:
|
|
// space_A full definition
|
|
// space_B declaration
|
|
// space_B definition
|
|
// struct_C full definition
|
|
// struct_D declaration
|
|
// struct_D definition
|
|
// union_E full definition
|
|
// union_F declaration
|
|
// union_F definition
|
|
// class_G full definition
|
|
// class_H declaration
|
|
// class_H definition
|
|
// func_I declaration
|
|
// func_I definition
|
|
//
|
|
// So:
|
|
// - space_A, struct_C, union_E, and class_G don't use specifications;
|
|
// - space_B, struct_D, union_F, and class_H do.
|
|
// - func_I uses a specification.
|
|
//
|
|
// The full name for func_I is thus:
|
|
//
|
|
// space_A::space_B::struct_C::struct_D::union_E::union_F::
|
|
// class_G::class_H::func_I
|
|
{
|
|
DIEHandler *space_A_handler
|
|
= StartNamedDIE(&root_handler_, dwarf2reader::DW_TAG_namespace,
|
|
"space_A");
|
|
DeclarationDIE(space_A_handler, 0x2e111126496596e2ULL,
|
|
dwarf2reader::DW_TAG_namespace, "space_B", "");
|
|
space_A_handler->Finish();
|
|
delete space_A_handler;
|
|
}
|
|
|
|
{
|
|
DIEHandler *space_B_handler
|
|
= StartSpecifiedDIE(&root_handler_, dwarf2reader::DW_TAG_namespace,
|
|
0x2e111126496596e2ULL);
|
|
DIEHandler *struct_C_handler
|
|
= StartNamedDIE(space_B_handler, dwarf2reader::DW_TAG_structure_type,
|
|
"struct_C");
|
|
DeclarationDIE(struct_C_handler, 0x20cd423bf2a25a4cULL,
|
|
dwarf2reader::DW_TAG_structure_type, "struct_D", "");
|
|
struct_C_handler->Finish();
|
|
delete struct_C_handler;
|
|
space_B_handler->Finish();
|
|
delete space_B_handler;
|
|
}
|
|
|
|
{
|
|
DIEHandler *struct_D_handler
|
|
= StartSpecifiedDIE(&root_handler_, dwarf2reader::DW_TAG_structure_type,
|
|
0x20cd423bf2a25a4cULL);
|
|
DIEHandler *union_E_handler
|
|
= StartNamedDIE(struct_D_handler, dwarf2reader::DW_TAG_union_type,
|
|
"union_E");
|
|
DeclarationDIE(union_E_handler, 0xe25c84805aa58c32ULL,
|
|
dwarf2reader::DW_TAG_union_type, "union_F", "");
|
|
union_E_handler->Finish();
|
|
delete union_E_handler;
|
|
struct_D_handler->Finish();
|
|
delete struct_D_handler;
|
|
}
|
|
|
|
{
|
|
DIEHandler *union_F_handler
|
|
= StartSpecifiedDIE(&root_handler_, dwarf2reader::DW_TAG_union_type,
|
|
0xe25c84805aa58c32ULL);
|
|
DIEHandler *class_G_handler
|
|
= StartNamedDIE(union_F_handler, dwarf2reader::DW_TAG_class_type,
|
|
"class_G");
|
|
DeclarationDIE(class_G_handler, 0xb70d960dcc173b6eULL,
|
|
dwarf2reader::DW_TAG_class_type, "class_H", "");
|
|
class_G_handler->Finish();
|
|
delete class_G_handler;
|
|
union_F_handler->Finish();
|
|
delete union_F_handler;
|
|
}
|
|
|
|
{
|
|
DIEHandler *class_H_handler
|
|
= StartSpecifiedDIE(&root_handler_, dwarf2reader::DW_TAG_class_type,
|
|
0xb70d960dcc173b6eULL);
|
|
DeclarationDIE(class_H_handler, 0x27ff829e3bf69f37ULL,
|
|
dwarf2reader::DW_TAG_subprogram, "func_I", "");
|
|
class_H_handler->Finish();
|
|
delete class_H_handler;
|
|
}
|
|
|
|
DefinitionDIE(&root_handler_, dwarf2reader::DW_TAG_subprogram,
|
|
0x27ff829e3bf69f37ULL, "",
|
|
0x5a0dd6bb85db754cULL, 0x3bccb213d08c7fd3ULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "space_A::space_B::struct_C::struct_D::union_E::union_F"
|
|
"::class_G::class_H::func_I",
|
|
0x5a0dd6bb85db754cULL, 0x3bccb213d08c7fd3ULL);
|
|
}
|
|
|
|
TEST_F(Specifications, InterCU) {
|
|
Module m("module-name", "module-os", "module-arch", "module-id");
|
|
DwarfCUToModule::FileContext fc("dwarf-filename", &m, true);
|
|
EXPECT_CALL(reporter_, UncoveredFunction(_)).WillOnce(Return());
|
|
MockLineToModuleHandler lr;
|
|
EXPECT_CALL(lr, ReadProgram(_,_,_,_)).Times(0);
|
|
|
|
// Kludge: satisfy reporter_'s expectation.
|
|
reporter_.SetCUName("compilation-unit-name");
|
|
|
|
// First CU. Declares class_A.
|
|
{
|
|
DwarfCUToModule root1_handler(&fc, &lr, &reporter_);
|
|
ASSERT_TRUE(root1_handler.StartCompilationUnit(0, 1, 2, 3, 3));
|
|
ASSERT_TRUE(root1_handler.StartRootDIE(1,
|
|
dwarf2reader::DW_TAG_compile_unit));
|
|
ProcessStrangeAttributes(&root1_handler);
|
|
ASSERT_TRUE(root1_handler.EndAttributes());
|
|
DeclarationDIE(&root1_handler, 0xb8fbfdd5f0b26fceULL,
|
|
dwarf2reader::DW_TAG_class_type, "class_A", "");
|
|
root1_handler.Finish();
|
|
}
|
|
|
|
// Second CU. Defines class_A, declares member_func_B.
|
|
{
|
|
DwarfCUToModule root2_handler(&fc, &lr, &reporter_);
|
|
ASSERT_TRUE(root2_handler.StartCompilationUnit(0, 1, 2, 3, 3));
|
|
ASSERT_TRUE(root2_handler.StartRootDIE(1,
|
|
dwarf2reader::DW_TAG_compile_unit));
|
|
ASSERT_TRUE(root2_handler.EndAttributes());
|
|
DIEHandler *class_A_handler
|
|
= StartSpecifiedDIE(&root2_handler, dwarf2reader::DW_TAG_class_type,
|
|
0xb8fbfdd5f0b26fceULL);
|
|
DeclarationDIE(class_A_handler, 0xb01fef8b380bd1a2ULL,
|
|
dwarf2reader::DW_TAG_subprogram, "member_func_B", "");
|
|
class_A_handler->Finish();
|
|
delete class_A_handler;
|
|
root2_handler.Finish();
|
|
}
|
|
|
|
// Third CU. Defines member_func_B.
|
|
{
|
|
DwarfCUToModule root3_handler(&fc, &lr, &reporter_);
|
|
ASSERT_TRUE(root3_handler.StartCompilationUnit(0, 1, 2, 3, 3));
|
|
ASSERT_TRUE(root3_handler.StartRootDIE(1,
|
|
dwarf2reader::DW_TAG_compile_unit));
|
|
ASSERT_TRUE(root3_handler.EndAttributes());
|
|
DefinitionDIE(&root3_handler, dwarf2reader::DW_TAG_subprogram,
|
|
0xb01fef8b380bd1a2ULL, "",
|
|
0x2618f00a1a711e53ULL, 0x4fd94b76d7c2caf5ULL);
|
|
root3_handler.Finish();
|
|
}
|
|
|
|
vector<Module::Function *> functions;
|
|
m.GetFunctions(&functions, functions.end());
|
|
EXPECT_EQ(1U, functions.size());
|
|
EXPECT_STREQ("class_A::member_func_B", functions[0]->name.c_str());
|
|
}
|
|
|
|
TEST_F(Specifications, UnhandledInterCU) {
|
|
Module m("module-name", "module-os", "module-arch", "module-id");
|
|
DwarfCUToModule::FileContext fc("dwarf-filename", &m, false);
|
|
EXPECT_CALL(reporter_, UncoveredFunction(_)).WillOnce(Return());
|
|
MockLineToModuleHandler lr;
|
|
EXPECT_CALL(lr, ReadProgram(_,_,_,_)).Times(0);
|
|
|
|
// Kludge: satisfy reporter_'s expectation.
|
|
reporter_.SetCUName("compilation-unit-name");
|
|
|
|
// First CU. Declares class_A.
|
|
{
|
|
DwarfCUToModule root1_handler(&fc, &lr, &reporter_);
|
|
ASSERT_TRUE(root1_handler.StartCompilationUnit(0, 1, 2, 3, 3));
|
|
ASSERT_TRUE(root1_handler.StartRootDIE(1,
|
|
dwarf2reader::DW_TAG_compile_unit));
|
|
ProcessStrangeAttributes(&root1_handler);
|
|
ASSERT_TRUE(root1_handler.EndAttributes());
|
|
DeclarationDIE(&root1_handler, 0xb8fbfdd5f0b26fceULL,
|
|
dwarf2reader::DW_TAG_class_type, "class_A", "");
|
|
root1_handler.Finish();
|
|
}
|
|
|
|
// Second CU. Defines class_A, declares member_func_B.
|
|
{
|
|
DwarfCUToModule root2_handler(&fc, &lr, &reporter_);
|
|
ASSERT_TRUE(root2_handler.StartCompilationUnit(0, 1, 2, 3, 3));
|
|
ASSERT_TRUE(root2_handler.StartRootDIE(1,
|
|
dwarf2reader::DW_TAG_compile_unit));
|
|
ASSERT_TRUE(root2_handler.EndAttributes());
|
|
EXPECT_CALL(reporter_, UnhandledInterCUReference(_, _)).Times(1);
|
|
DIEHandler *class_A_handler
|
|
= StartSpecifiedDIE(&root2_handler, dwarf2reader::DW_TAG_class_type,
|
|
0xb8fbfdd5f0b26fceULL);
|
|
DeclarationDIE(class_A_handler, 0xb01fef8b380bd1a2ULL,
|
|
dwarf2reader::DW_TAG_subprogram, "member_func_B", "");
|
|
class_A_handler->Finish();
|
|
delete class_A_handler;
|
|
root2_handler.Finish();
|
|
}
|
|
|
|
// Third CU. Defines member_func_B.
|
|
{
|
|
DwarfCUToModule root3_handler(&fc, &lr, &reporter_);
|
|
ASSERT_TRUE(root3_handler.StartCompilationUnit(0, 1, 2, 3, 3));
|
|
ASSERT_TRUE(root3_handler.StartRootDIE(1,
|
|
dwarf2reader::DW_TAG_compile_unit));
|
|
ASSERT_TRUE(root3_handler.EndAttributes());
|
|
EXPECT_CALL(reporter_, UnhandledInterCUReference(_, _)).Times(1);
|
|
EXPECT_CALL(reporter_, UnnamedFunction(_)).Times(1);
|
|
DefinitionDIE(&root3_handler, dwarf2reader::DW_TAG_subprogram,
|
|
0xb01fef8b380bd1a2ULL, "",
|
|
0x2618f00a1a711e53ULL, 0x4fd94b76d7c2caf5ULL);
|
|
root3_handler.Finish();
|
|
}
|
|
}
|
|
|
|
TEST_F(Specifications, BadOffset) {
|
|
PushLine(0xa0277efd7ce83771ULL, 0x149554a184c730c1ULL, "line-file", 56636272);
|
|
EXPECT_CALL(reporter_, UnknownSpecification(_, 0x2be953efa6f9a996ULL))
|
|
.WillOnce(Return());
|
|
|
|
StartCU();
|
|
DeclarationDIE(&root_handler_, 0xefd7f7752c27b7e4ULL,
|
|
dwarf2reader::DW_TAG_subprogram, "", "");
|
|
DefinitionDIE(&root_handler_, dwarf2reader::DW_TAG_subprogram,
|
|
0x2be953efa6f9a996ULL, "function",
|
|
0xa0277efd7ce83771ULL, 0x149554a184c730c1ULL);
|
|
root_handler_.Finish();
|
|
}
|
|
|
|
TEST_F(Specifications, FunctionDefinitionHasOwnName) {
|
|
PushLine(0xced50b3eea81022cULL, 0x08dd4d301cc7a7d2ULL, "line-file", 56792403);
|
|
|
|
StartCU();
|
|
DeclarationDIE(&root_handler_, 0xc34ff4786cae78bdULL,
|
|
dwarf2reader::DW_TAG_subprogram, "declaration-name", "");
|
|
DefinitionDIE(&root_handler_, dwarf2reader::DW_TAG_subprogram,
|
|
0xc34ff4786cae78bdULL, "definition-name",
|
|
0xced50b3eea81022cULL, 0x08dd4d301cc7a7d2ULL);
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "definition-name",
|
|
0xced50b3eea81022cULL, 0x08dd4d301cc7a7d2ULL);
|
|
}
|
|
|
|
TEST_F(Specifications, ClassDefinitionHasOwnName) {
|
|
PushLine(0x1d0f5e0f6ce309bdULL, 0x654e1852ec3599e7ULL, "line-file", 57119241);
|
|
|
|
StartCU();
|
|
DeclarationDIE(&root_handler_, 0xd0fe467ec2f1a58cULL,
|
|
dwarf2reader::DW_TAG_class_type, "class-declaration-name", "");
|
|
|
|
dwarf2reader::DIEHandler *class_definition
|
|
= StartSpecifiedDIE(&root_handler_, dwarf2reader::DW_TAG_class_type,
|
|
0xd0fe467ec2f1a58cULL, "class-definition-name");
|
|
ASSERT_TRUE(class_definition);
|
|
DeclarationDIE(class_definition, 0x6d028229c15623dbULL,
|
|
dwarf2reader::DW_TAG_subprogram,
|
|
"function-declaration-name", "");
|
|
class_definition->Finish();
|
|
delete class_definition;
|
|
|
|
DefinitionDIE(&root_handler_, dwarf2reader::DW_TAG_subprogram,
|
|
0x6d028229c15623dbULL, "function-definition-name",
|
|
0x1d0f5e0f6ce309bdULL, 0x654e1852ec3599e7ULL);
|
|
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "class-definition-name::function-definition-name",
|
|
0x1d0f5e0f6ce309bdULL, 0x654e1852ec3599e7ULL);
|
|
}
|
|
|
|
// DIEs that cite a specification should prefer the specification's
|
|
// parents over their own when choosing qualified names. In this test,
|
|
// we take the name from our definition but the enclosing scope name
|
|
// from our declaration. I don't see why they'd ever be different, but
|
|
// we want to verify what DwarfCUToModule is looking at.
|
|
TEST_F(Specifications, PreferSpecificationParents) {
|
|
PushLine(0xbbd9d54dce3b95b7ULL, 0x39188b7b52b0899fULL, "line-file", 79488694);
|
|
|
|
StartCU();
|
|
{
|
|
dwarf2reader::DIEHandler *declaration_class_handler =
|
|
StartNamedDIE(&root_handler_, dwarf2reader::DW_TAG_class_type,
|
|
"declaration-class");
|
|
DeclarationDIE(declaration_class_handler, 0x9ddb35517455ef7aULL,
|
|
dwarf2reader::DW_TAG_subprogram, "function-declaration",
|
|
"");
|
|
declaration_class_handler->Finish();
|
|
delete declaration_class_handler;
|
|
}
|
|
{
|
|
dwarf2reader::DIEHandler *definition_class_handler
|
|
= StartNamedDIE(&root_handler_, dwarf2reader::DW_TAG_class_type,
|
|
"definition-class");
|
|
DefinitionDIE(definition_class_handler, dwarf2reader::DW_TAG_subprogram,
|
|
0x9ddb35517455ef7aULL, "function-definition",
|
|
0xbbd9d54dce3b95b7ULL, 0x39188b7b52b0899fULL);
|
|
definition_class_handler->Finish();
|
|
delete definition_class_handler;
|
|
}
|
|
root_handler_.Finish();
|
|
|
|
TestFunctionCount(1);
|
|
TestFunction(0, "declaration-class::function-definition",
|
|
0xbbd9d54dce3b95b7ULL, 0x39188b7b52b0899fULL);
|
|
}
|
|
|
|
class CUErrors: public CUFixtureBase, public Test { };
|
|
|
|
TEST_F(CUErrors, BadStmtList) {
|
|
EXPECT_CALL(reporter_, BadLineInfoOffset(dummy_line_size_ + 10)).Times(1);
|
|
|
|
ASSERT_TRUE(root_handler_
|
|
.StartCompilationUnit(0xc591d5b037543d7cULL, 0x11, 0xcd,
|
|
0x2d7d19546cf6590cULL, 3));
|
|
ASSERT_TRUE(root_handler_.StartRootDIE(0xae789dc102cfca54ULL,
|
|
dwarf2reader::DW_TAG_compile_unit));
|
|
root_handler_.ProcessAttributeString(dwarf2reader::DW_AT_name,
|
|
dwarf2reader::DW_FORM_strp,
|
|
"compilation-unit-name");
|
|
root_handler_.ProcessAttributeUnsigned(dwarf2reader::DW_AT_stmt_list,
|
|
dwarf2reader::DW_FORM_ref4,
|
|
dummy_line_size_ + 10);
|
|
root_handler_.EndAttributes();
|
|
root_handler_.Finish();
|
|
}
|
|
|
|
TEST_F(CUErrors, NoLineSection) {
|
|
EXPECT_CALL(reporter_, MissingSection(".debug_line")).Times(1);
|
|
PushLine(0x88507fb678052611ULL, 0x42c8e9de6bbaa0faULL, "line-file", 64472290);
|
|
// Delete the entry for .debug_line added by the fixture class's constructor.
|
|
file_context_.ClearSectionMapForTest();
|
|
|
|
StartCU();
|
|
root_handler_.Finish();
|
|
}
|
|
|
|
TEST_F(CUErrors, BadDwarfVersion1) {
|
|
// Kludge: satisfy reporter_'s expectation.
|
|
reporter_.SetCUName("compilation-unit-name");
|
|
|
|
ASSERT_FALSE(root_handler_
|
|
.StartCompilationUnit(0xadf6e0eb71e2b0d9ULL, 0x4d, 0x90,
|
|
0xc9de224ccb99ac3eULL, 1));
|
|
}
|
|
|
|
TEST_F(CUErrors, GoodDwarfVersion2) {
|
|
// Kludge: satisfy reporter_'s expectation.
|
|
reporter_.SetCUName("compilation-unit-name");
|
|
|
|
ASSERT_TRUE(root_handler_
|
|
.StartCompilationUnit(0xadf6e0eb71e2b0d9ULL, 0x4d, 0x90,
|
|
0xc9de224ccb99ac3eULL, 2));
|
|
}
|
|
|
|
TEST_F(CUErrors, GoodDwarfVersion3) {
|
|
// Kludge: satisfy reporter_'s expectation.
|
|
reporter_.SetCUName("compilation-unit-name");
|
|
|
|
ASSERT_TRUE(root_handler_
|
|
.StartCompilationUnit(0xadf6e0eb71e2b0d9ULL, 0x4d, 0x90,
|
|
0xc9de224ccb99ac3eULL, 3));
|
|
}
|
|
|
|
TEST_F(CUErrors, BadCURootDIETag) {
|
|
// Kludge: satisfy reporter_'s expectation.
|
|
reporter_.SetCUName("compilation-unit-name");
|
|
|
|
ASSERT_TRUE(root_handler_
|
|
.StartCompilationUnit(0xadf6e0eb71e2b0d9ULL, 0x4d, 0x90,
|
|
0xc9de224ccb99ac3eULL, 3));
|
|
|
|
ASSERT_FALSE(root_handler_.StartRootDIE(0x02e56bfbda9e7337ULL,
|
|
dwarf2reader::DW_TAG_subprogram));
|
|
}
|
|
|
|
// Tests for DwarfCUToModule::Reporter. These just produce (or fail to
|
|
// produce) output, so their results need to be checked by hand.
|
|
struct Reporter: public Test {
|
|
Reporter()
|
|
: reporter("filename", 0x123456789abcdef0ULL),
|
|
function("function name", 0x19c45c30770c1eb0ULL),
|
|
file("source file name") {
|
|
reporter.SetCUName("compilation-unit-name");
|
|
|
|
function.size = 0x89808a5bdfa0a6a3ULL;
|
|
function.parameter_size = 0x6a329f18683dcd51ULL;
|
|
|
|
line.address = 0x3606ac6267aebeccULL;
|
|
line.size = 0x5de482229f32556aULL;
|
|
line.file = &file;
|
|
line.number = 93400201;
|
|
}
|
|
|
|
DwarfCUToModule::WarningReporter reporter;
|
|
Module::Function function;
|
|
Module::File file;
|
|
Module::Line line;
|
|
};
|
|
|
|
TEST_F(Reporter, UnknownSpecification) {
|
|
reporter.UnknownSpecification(0x123456789abcdef1ULL, 0x323456789abcdef2ULL);
|
|
}
|
|
|
|
TEST_F(Reporter, UnknownAbstractOrigin) {
|
|
reporter.UnknownAbstractOrigin(0x123456789abcdef1ULL, 0x323456789abcdef2ULL);
|
|
}
|
|
|
|
TEST_F(Reporter, MissingSection) {
|
|
reporter.MissingSection("section name");
|
|
}
|
|
|
|
TEST_F(Reporter, BadLineInfoOffset) {
|
|
reporter.BadLineInfoOffset(0x123456789abcdef1ULL);
|
|
}
|
|
|
|
TEST_F(Reporter, UncoveredFunctionDisabled) {
|
|
reporter.UncoveredFunction(function);
|
|
EXPECT_FALSE(reporter.uncovered_warnings_enabled());
|
|
}
|
|
|
|
TEST_F(Reporter, UncoveredFunctionEnabled) {
|
|
reporter.set_uncovered_warnings_enabled(true);
|
|
reporter.UncoveredFunction(function);
|
|
EXPECT_TRUE(reporter.uncovered_warnings_enabled());
|
|
}
|
|
|
|
TEST_F(Reporter, UncoveredLineDisabled) {
|
|
reporter.UncoveredLine(line);
|
|
EXPECT_FALSE(reporter.uncovered_warnings_enabled());
|
|
}
|
|
|
|
TEST_F(Reporter, UncoveredLineEnabled) {
|
|
reporter.set_uncovered_warnings_enabled(true);
|
|
reporter.UncoveredLine(line);
|
|
EXPECT_TRUE(reporter.uncovered_warnings_enabled());
|
|
}
|
|
|
|
TEST_F(Reporter, UnnamedFunction) {
|
|
reporter.UnnamedFunction(0x90c0baff9dedb2d9ULL);
|
|
}
|
|
|
|
// Would be nice to also test:
|
|
// - overlapping lines, functions
|