2015-02-20 21:07:41 +08:00
//===-- MICmdCmdSymbol.cpp --------------------------------------*- C++ -*-===//
//
// The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
// Overview: CMICmdCmdSymbolListLines implementation.
// Third Party Headers:
2015-02-21 08:39:13 +08:00
# include "lldb/API/SBCommandInterpreter.h"
2016-09-05 23:15:12 +08:00
# include "llvm/ADT/SmallVector.h"
# include "llvm/ADT/StringRef.h"
# include "llvm/Support/Regex.h"
2015-02-20 21:07:41 +08:00
// In-house headers:
# include "MICmdArgValFile.h"
# include "MICmdCmdSymbol.h"
# include "MICmnLLDBDebugSessionInfo.h"
# include "MICmnMIResultRecord.h"
# include "MICmnMIValueList.h"
# include "MICmnMIValueTuple.h"
//++ ------------------------------------------------------------------------------------
// Details: CMICmdCmdSymbolListLines constructor.
// Type: Method.
// Args: None.
// Return: None.
// Throws: None.
//--
2015-08-04 18:24:20 +08:00
CMICmdCmdSymbolListLines : : CMICmdCmdSymbolListLines ( )
2015-02-20 21:07:41 +08:00
: m_constStrArgNameFile ( " file " )
{
// Command factory matches this name with that received from the stdin stream
m_strMiCmd = " symbol-list-lines " ;
// Required by the CMICmdFactory when registering *this command
m_pSelfCreatorFn = & CMICmdCmdSymbolListLines : : CreateSelf ;
}
//++ ------------------------------------------------------------------------------------
// Details: CMICmdCmdSymbolListLines destructor.
// Type: Overrideable.
// Args: None.
// Return: None.
// Throws: None.
//--
2015-08-04 18:24:20 +08:00
CMICmdCmdSymbolListLines : : ~ CMICmdCmdSymbolListLines ( )
2015-02-20 21:07:41 +08:00
{
}
//++ ------------------------------------------------------------------------------------
// Details: The invoker requires this function. The parses the command line options
// arguments to extract values for each of those arguments.
// Type: Overridden.
// Args: None.
// Return: MIstatus::success - Functional succeeded.
// MIstatus::failure - Functional failed.
// Throws: None.
//--
bool
2015-08-04 18:24:20 +08:00
CMICmdCmdSymbolListLines : : ParseArgs ( )
2015-02-20 21:07:41 +08:00
{
2015-09-15 20:00:08 +08:00
m_setCmdArgs . Add ( new CMICmdArgValFile ( m_constStrArgNameFile , true , true ) ) ;
2015-07-23 01:07:27 +08:00
return ParseValidateCmdOptions ( ) ;
2015-02-20 21:07:41 +08:00
}
//++ ------------------------------------------------------------------------------------
// Details: The invoker requires this function. The command does work in this function.
// The command is likely to communicate with the LLDB SBDebugger in here.
// Synopsis: -symbol-list-lines file
// Ref: http://sourceware.org/gdb/onlinedocs/gdb/GDB_002fMI-Symbol-Query.html#GDB_002fMI-Symbol-Query
// Type: Overridden.
// Args: None.
// Return: MIstatus::success - Functional succeeded.
// MIstatus::failure - Functional failed.
// Throws: None.
//--
bool
2015-08-04 18:24:20 +08:00
CMICmdCmdSymbolListLines : : Execute ( )
2015-02-20 21:07:41 +08:00
{
CMICMDBASE_GETOPTION ( pArgFile , File , m_constStrArgNameFile ) ;
const CMIUtilString & strFilePath ( pArgFile - > GetValue ( ) ) ;
2016-01-06 03:51:51 +08:00
const CMIUtilString strCmd ( CMIUtilString : : Format ( " source info --file \" %s \" " , strFilePath . AddSlashes ( ) . c_str ( ) ) ) ;
2015-02-20 21:07:41 +08:00
CMICmnLLDBDebugSessionInfo & rSessionInfo ( CMICmnLLDBDebugSessionInfo : : Instance ( ) ) ;
const lldb : : ReturnStatus rtn = rSessionInfo . GetDebugger ( ) . GetCommandInterpreter ( ) . HandleCommand ( strCmd . c_str ( ) , m_lldbResult ) ;
MIunused ( rtn ) ;
return MIstatus : : success ;
}
2015-09-18 00:22:30 +08:00
//++ ------------------------------------------------------------------------------------
// Details: Helper function for parsing the header returned from lldb for the command:
// target modules dump line-table <file>
// where the header is of the format:
// Line table for /path/to/file in `/path/to/module
// Args: input - (R) Input string to parse.
// file - (W) String representing the file.
// Return: bool - True = input was parsed successfully, false = input could not be parsed.
// Throws: None.
//--
static bool
ParseLLDBLineAddressHeader ( const char * input , CMIUtilString & file )
{
// Match LineEntry using regex.
2016-09-05 23:15:12 +08:00
static llvm : : Regex g_lineentry_header_regex (
llvm : : StringRef ( " ^ *Lines found for file (.+) in compilation unit (.+) in `(.+)$ " ) ) ;
// ^1=file ^2=cu ^3=module
2015-09-18 00:22:30 +08:00
2016-09-05 23:15:12 +08:00
llvm : : SmallVector < llvm : : StringRef , 4 > match ;
2015-09-18 00:22:30 +08:00
2016-09-05 23:15:12 +08:00
const bool ok = g_lineentry_header_regex . match ( input , & match ) ;
2015-09-18 00:22:30 +08:00
if ( ok )
2016-09-05 23:15:12 +08:00
file = match [ 1 ] ;
2015-09-18 00:22:30 +08:00
return ok ;
}
//++ ------------------------------------------------------------------------------------
// Details: Helper function for parsing a line entry returned from lldb for the command:
// target modules dump line-table <file>
// where the line entry is of the format:
// 0x0000000100000e70: /path/to/file:3002[:4]
// addr file line column(opt)
// Args: input - (R) Input string to parse.
// addr - (W) String representing the pc address.
// file - (W) String representing the file.
// line - (W) String representing the line.
// Return: bool - True = input was parsed successfully, false = input could not be parsed.
// Throws: None.
//--
static bool
ParseLLDBLineAddressEntry ( const char * input , CMIUtilString & addr ,
CMIUtilString & file , CMIUtilString & line )
{
// Note: Ambiguities arise because the column is optional, and
// because : can appear in filenames or as a byte in a multibyte
// UTF8 character. We keep those cases to a minimum by using regex
// to work on the string from both the left and right, so that what
// is remains is assumed to be the filename.
// Match LineEntry using regex.
2016-09-05 23:15:12 +08:00
static llvm : : Regex g_lineentry_nocol_regex (
llvm : : StringRef ( " ^ * \\ [(0x[0-9a-fA-F]+)-(0x[0-9a-fA-F]+) \\ ): (.+):([0-9]+)$ " ) ) ;
static llvm : : Regex g_lineentry_col_regex (
llvm : : StringRef ( " ^ * \\ [(0x[0-9a-fA-F]+)-(0x[0-9a-fA-F]+) \\ ): (.+):([0-9]+):[0-9]+$ " ) ) ;
// ^1=start ^2=end ^3=f ^4=line ^5=:col(opt)
2015-09-18 00:22:30 +08:00
2016-09-05 23:15:12 +08:00
llvm : : SmallVector < llvm : : StringRef , 6 > match ;
2015-09-18 00:22:30 +08:00
// First try matching the LineEntry with the column,
// then try without the column.
2016-09-05 23:15:12 +08:00
const bool ok = g_lineentry_col_regex . match ( input , & match ) | |
g_lineentry_nocol_regex . match ( input , & match ) ;
2015-09-18 00:22:30 +08:00
if ( ok )
{
2016-09-05 23:15:12 +08:00
addr = match [ 1 ] ;
file = match [ 3 ] ;
line = match [ 4 ] ;
2015-09-18 00:22:30 +08:00
}
return ok ;
}
2015-02-20 21:07:41 +08:00
//++ ------------------------------------------------------------------------------------
// Details: The invoker requires this function. The command prepares a MI Record Result
// for the work carried out in the Execute().
// Type: Overridden.
// Args: None.
// Return: MIstatus::success - Functional succeeded.
// MIstatus::failure - Functional failed.
// Throws: None.
//--
bool
2015-08-04 18:24:20 +08:00
CMICmdCmdSymbolListLines : : Acknowledge ( )
2015-02-20 21:07:41 +08:00
{
if ( m_lldbResult . GetErrorSize ( ) > 0 )
{
2015-07-03 21:45:34 +08:00
const char * pLldbErr = m_lldbResult . GetError ( ) ;
2015-02-20 21:07:41 +08:00
const CMIUtilString strMsg ( CMIUtilString ( pLldbErr ) . StripCRAll ( ) ) ;
const CMICmnMIValueConst miValueConst ( strMsg ) ;
const CMICmnMIValueResult miValueResult ( " message " , miValueConst ) ;
const CMICmnMIResultRecord miRecordResult ( m_cmdData . strMiCmdToken , CMICmnMIResultRecord : : eResultClass_Error , miValueResult ) ;
m_miResultRecord = miRecordResult ;
}
else
{
CMIUtilString : : VecString_t vecLines ;
const CMIUtilString strLldbMsg ( m_lldbResult . GetOutput ( ) ) ;
const MIuint nLines ( strLldbMsg . SplitLines ( vecLines ) ) ;
2015-09-18 00:22:30 +08:00
// Parse the file from the header.
const CMIUtilString & rWantFile ( vecLines [ 0 ] ) ;
CMIUtilString strWantFile ;
if ( ! ParseLLDBLineAddressHeader ( rWantFile . c_str ( ) , strWantFile ) )
{
// Unexpected error - parsing failed.
// MI print "%s^error,msg=\"Command '-symbol-list-lines'. Error: Line address header is absent or has an unknown format.\""
const CMICmnMIValueConst miValueConst ( CMIUtilString : : Format ( MIRSRC ( IDS_CMD_ERR_SOME_ERROR ) , m_cmdData . strMiCmd . c_str ( ) , " Line address header is absent or has an unknown format. " ) ) ;
const CMICmnMIValueResult miValueResult ( " msg " , miValueConst ) ;
const CMICmnMIResultRecord miRecordResult ( m_cmdData . strMiCmdToken , CMICmnMIResultRecord : : eResultClass_Error , miValueResult ) ;
m_miResultRecord = miRecordResult ;
return MIstatus : : success ;
}
// Parse the line address entries.
2015-02-20 21:07:41 +08:00
CMICmnMIValueList miValueList ( true ) ;
for ( MIuint i = 1 ; i < nLines ; + + i )
{
// String looks like:
// 0x0000000100000e70: /path/to/file:3[:4]
const CMIUtilString & rLine ( vecLines [ i ] ) ;
2015-09-18 00:22:30 +08:00
CMIUtilString strAddr ;
CMIUtilString strFile ;
CMIUtilString strLine ;
if ( ! ParseLLDBLineAddressEntry ( rLine . c_str ( ) , strAddr , strFile , strLine ) )
continue ;
2015-02-20 21:07:41 +08:00
const CMICmnMIValueConst miValueConst ( strAddr ) ;
const CMICmnMIValueResult miValueResult ( " pc " , miValueConst ) ;
CMICmnMIValueTuple miValueTuple ( miValueResult ) ;
const CMICmnMIValueConst miValueConst2 ( strLine ) ;
const CMICmnMIValueResult miValueResult2 ( " line " , miValueConst2 ) ;
2015-07-23 01:07:27 +08:00
miValueTuple . Add ( miValueResult2 ) ;
2015-02-20 21:07:41 +08:00
2015-07-23 01:07:27 +08:00
miValueList . Add ( miValueTuple ) ;
2015-02-20 21:07:41 +08:00
}
// MI print "%s^done,lines=[{pc=\"%d\",line=\"%d\"}...]"
const CMICmnMIValueResult miValueResult ( " lines " , miValueList ) ;
const CMICmnMIResultRecord miRecordResult ( m_cmdData . strMiCmdToken , CMICmnMIResultRecord : : eResultClass_Done , miValueResult ) ;
m_miResultRecord = miRecordResult ;
}
return MIstatus : : success ;
}
//++ ------------------------------------------------------------------------------------
// Details: Required by the CMICmdFactory when registering *this command. The factory
// calls this function to create an instance of *this command.
// Type: Static method.
// Args: None.
// Return: CMICmdBase * - Pointer to a new command.
// Throws: None.
//--
CMICmdBase *
2015-08-04 18:24:20 +08:00
CMICmdCmdSymbolListLines : : CreateSelf ( )
2015-02-20 21:07:41 +08:00
{
return new CMICmdCmdSymbolListLines ( ) ;
}