130fdc8d8SChris Lattner //===-- BreakpointResolverFileLine.cpp --------------------------*- C++ -*-===//
230fdc8d8SChris Lattner //
330fdc8d8SChris Lattner //                     The LLVM Compiler Infrastructure
430fdc8d8SChris Lattner //
530fdc8d8SChris Lattner // This file is distributed under the University of Illinois Open Source
630fdc8d8SChris Lattner // License. See LICENSE.TXT for details.
730fdc8d8SChris Lattner //
830fdc8d8SChris Lattner //===----------------------------------------------------------------------===//
930fdc8d8SChris Lattner 
1030fdc8d8SChris Lattner #include "lldb/Breakpoint/BreakpointResolverFileLine.h"
1130fdc8d8SChris Lattner 
1230fdc8d8SChris Lattner // C Includes
1330fdc8d8SChris Lattner // C++ Includes
1430fdc8d8SChris Lattner // Other libraries and framework includes
1530fdc8d8SChris Lattner // Project includes
1630fdc8d8SChris Lattner #include "lldb/Breakpoint/BreakpointLocation.h"
1730fdc8d8SChris Lattner #include "lldb/Core/Log.h"
181f746071SGreg Clayton #include "lldb/Core/Module.h"
1930fdc8d8SChris Lattner #include "lldb/Core/StreamString.h"
201f746071SGreg Clayton #include "lldb/Symbol/CompileUnit.h"
211f746071SGreg Clayton #include "lldb/Symbol/Function.h"
2230fdc8d8SChris Lattner 
2330fdc8d8SChris Lattner using namespace lldb;
2430fdc8d8SChris Lattner using namespace lldb_private;
2530fdc8d8SChris Lattner 
2630fdc8d8SChris Lattner //----------------------------------------------------------------------
2730fdc8d8SChris Lattner // BreakpointResolverFileLine:
2830fdc8d8SChris Lattner //----------------------------------------------------------------------
29*b9c1b51eSKate Stone BreakpointResolverFileLine::BreakpointResolverFileLine(
30*b9c1b51eSKate Stone     Breakpoint *bkpt, const FileSpec &file_spec, uint32_t line_no,
31*b9c1b51eSKate Stone     lldb::addr_t offset, bool check_inlines, bool skip_prologue,
32*b9c1b51eSKate Stone     bool exact_match)
33*b9c1b51eSKate Stone     : BreakpointResolver(bkpt, BreakpointResolver::FileLineResolver, offset),
34*b9c1b51eSKate Stone       m_file_spec(file_spec), m_line_number(line_no), m_inlines(check_inlines),
35*b9c1b51eSKate Stone       m_skip_prologue(skip_prologue), m_exact_match(exact_match) {}
3630fdc8d8SChris Lattner 
37*b9c1b51eSKate Stone BreakpointResolverFileLine::~BreakpointResolverFileLine() {}
3830fdc8d8SChris Lattner 
3930fdc8d8SChris Lattner Searcher::CallbackReturn
40*b9c1b51eSKate Stone BreakpointResolverFileLine::SearchCallback(SearchFilter &filter,
4130fdc8d8SChris Lattner                                            SymbolContext &context,
42*b9c1b51eSKate Stone                                            Address *addr, bool containing) {
4330fdc8d8SChris Lattner   SymbolContextList sc_list;
4430fdc8d8SChris Lattner 
4530fdc8d8SChris Lattner   assert(m_breakpoint != NULL);
4630fdc8d8SChris Lattner 
47*b9c1b51eSKate Stone   // There is a tricky bit here.  You can have two compilation units that
48*b9c1b51eSKate Stone   // #include the same file, and
49*b9c1b51eSKate Stone   // in one of them the function at m_line_number is used (and so code and a
50*b9c1b51eSKate Stone   // line entry for it is generated) but in the
51*b9c1b51eSKate Stone   // other it isn't.  If we considered the CU's independently, then in the
52*b9c1b51eSKate Stone   // second inclusion, we'd move the breakpoint
53*b9c1b51eSKate Stone   // to the next function that actually generated code in the header file.  That
54*b9c1b51eSKate Stone   // would end up being confusing.
55*b9c1b51eSKate Stone   // So instead, we do the CU iterations by hand here, then scan through the
56*b9c1b51eSKate Stone   // complete list of matches, and figure out
57bc2f9182SJim Ingham   // the closest line number match, and only set breakpoints on that match.
58bc2f9182SJim Ingham 
59*b9c1b51eSKate Stone   // Note also that if file_spec only had a file name and not a directory, there
60*b9c1b51eSKate Stone   // may be many different file spec's in
61*b9c1b51eSKate Stone   // the resultant list.  The closest line match for one will not be right for
62*b9c1b51eSKate Stone   // some totally different file.
63*b9c1b51eSKate Stone   // So we go through the match list and pull out the sets that have the same
64*b9c1b51eSKate Stone   // file spec in their line_entry
65bc2f9182SJim Ingham   // and treat each set separately.
66bc2f9182SJim Ingham 
67c7bece56SGreg Clayton   const size_t num_comp_units = context.module_sp->GetNumCompileUnits();
68*b9c1b51eSKate Stone   for (size_t i = 0; i < num_comp_units; i++) {
69bc2f9182SJim Ingham     CompUnitSP cu_sp(context.module_sp->GetCompileUnitAtIndex(i));
70*b9c1b51eSKate Stone     if (cu_sp) {
712dafd8edSGreg Clayton       if (filter.CompUnitPasses(*cu_sp))
72*b9c1b51eSKate Stone         cu_sp->ResolveSymbolContext(m_file_spec, m_line_number, m_inlines,
73*b9c1b51eSKate Stone                                     m_exact_match, eSymbolContextEverything,
74*b9c1b51eSKate Stone                                     sc_list);
75bc2f9182SJim Ingham     }
762dafd8edSGreg Clayton   }
7730fdc8d8SChris Lattner   StreamString s;
78*b9c1b51eSKate Stone   s.Printf("for %s:%d ", m_file_spec.GetFilename().AsCString("<Unknown>"),
79969795f1SJim Ingham            m_line_number);
80f642373cSJim Ingham 
81f642373cSJim Ingham   SetSCMatchesByLine(filter, sc_list, m_skip_prologue, s.GetData());
82bc2f9182SJim Ingham 
8330fdc8d8SChris Lattner   return Searcher::eCallbackReturnContinue;
8430fdc8d8SChris Lattner }
8530fdc8d8SChris Lattner 
86*b9c1b51eSKate Stone Searcher::Depth BreakpointResolverFileLine::GetDepth() {
87bc2f9182SJim Ingham   return Searcher::eDepthModule;
8830fdc8d8SChris Lattner }
8930fdc8d8SChris Lattner 
90*b9c1b51eSKate Stone void BreakpointResolverFileLine::GetDescription(Stream *s) {
91*b9c1b51eSKate Stone   s->Printf("file = '%s', line = %u, exact_match = %d",
92*b9c1b51eSKate Stone             m_file_spec.GetPath().c_str(), m_line_number, m_exact_match);
9330fdc8d8SChris Lattner }
9430fdc8d8SChris Lattner 
95*b9c1b51eSKate Stone void BreakpointResolverFileLine::Dump(Stream *s) const {}
9630fdc8d8SChris Lattner 
9733df7cd3SJim Ingham lldb::BreakpointResolverSP
98*b9c1b51eSKate Stone BreakpointResolverFileLine::CopyForBreakpoint(Breakpoint &breakpoint) {
99*b9c1b51eSKate Stone   lldb::BreakpointResolverSP ret_sp(new BreakpointResolverFileLine(
100*b9c1b51eSKate Stone       &breakpoint, m_file_spec, m_line_number, m_offset, m_inlines,
101*b9c1b51eSKate Stone       m_skip_prologue, m_exact_match));
10233df7cd3SJim Ingham 
10333df7cd3SJim Ingham   return ret_sp;
10433df7cd3SJim Ingham }
105