2010-08-28 07:31:21 +08:00
|
|
|
//===-- ClangUserExpression.cpp -------------------------------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
// C Includes
|
|
|
|
#include <stdio.h>
|
|
|
|
#if HAVE_SYS_TYPES_H
|
|
|
|
# include <sys/types.h>
|
|
|
|
#endif
|
|
|
|
|
|
|
|
// C++ Includes
|
|
|
|
|
|
|
|
#include "lldb/Core/ConstString.h"
|
2011-05-24 05:40:23 +08:00
|
|
|
#include "lldb/Core/Log.h"
|
2010-08-28 07:31:21 +08:00
|
|
|
#include "lldb/Core/Stream.h"
|
2011-01-19 03:36:39 +08:00
|
|
|
#include "lldb/Core/StreamFile.h"
|
2010-09-14 05:34:21 +08:00
|
|
|
#include "lldb/Expression/ClangExpressionDeclMap.h"
|
2010-08-28 07:31:21 +08:00
|
|
|
#include "lldb/Expression/ClangExpressionParser.h"
|
|
|
|
#include "lldb/Expression/ClangUtilityFunction.h"
|
|
|
|
#include "lldb/Host/Host.h"
|
|
|
|
#include "lldb/Target/ExecutionContext.h"
|
|
|
|
#include "lldb/Target/Target.h"
|
|
|
|
|
|
|
|
using namespace lldb_private;
|
|
|
|
|
|
|
|
//------------------------------------------------------------------
|
|
|
|
/// Constructor
|
|
|
|
///
|
|
|
|
/// @param[in] text
|
|
|
|
/// The text of the function. Must be a full translation unit.
|
|
|
|
///
|
|
|
|
/// @param[in] name
|
|
|
|
/// The name of the function, as used in the text.
|
|
|
|
//------------------------------------------------------------------
|
|
|
|
ClangUtilityFunction::ClangUtilityFunction (const char *text,
|
|
|
|
const char *name) :
|
2011-01-20 07:00:49 +08:00
|
|
|
ClangExpression (),
|
|
|
|
m_function_text (text),
|
|
|
|
m_function_name (name)
|
2010-08-28 07:31:21 +08:00
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2010-09-25 07:07:41 +08:00
|
|
|
ClangUtilityFunction::~ClangUtilityFunction ()
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
2010-08-28 07:31:21 +08:00
|
|
|
//------------------------------------------------------------------
|
|
|
|
/// Install the utility function into a process
|
|
|
|
///
|
|
|
|
/// @param[in] error_stream
|
|
|
|
/// A stream to print parse errors and warnings to.
|
|
|
|
///
|
|
|
|
/// @param[in] exe_ctx
|
|
|
|
/// The execution context to install the utility function to.
|
|
|
|
///
|
|
|
|
/// @return
|
|
|
|
/// True on success (no errors); false otherwise.
|
|
|
|
//------------------------------------------------------------------
|
|
|
|
bool
|
|
|
|
ClangUtilityFunction::Install (Stream &error_stream,
|
|
|
|
ExecutionContext &exe_ctx)
|
2010-09-01 08:58:00 +08:00
|
|
|
{
|
2011-05-24 05:40:23 +08:00
|
|
|
lldb::LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_EXPRESSIONS));
|
|
|
|
|
2011-01-20 07:00:49 +08:00
|
|
|
if (m_jit_start_addr != LLDB_INVALID_ADDRESS)
|
2010-09-01 08:58:00 +08:00
|
|
|
{
|
|
|
|
error_stream.PutCString("error: already installed\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2010-08-28 07:31:21 +08:00
|
|
|
////////////////////////////////////
|
|
|
|
// Set up the target and compiler
|
|
|
|
//
|
|
|
|
|
|
|
|
Target *target = exe_ctx.target;
|
|
|
|
|
|
|
|
if (!target)
|
|
|
|
{
|
|
|
|
error_stream.PutCString ("error: invalid target\n");
|
|
|
|
return false;
|
|
|
|
}
|
2011-05-24 05:40:23 +08:00
|
|
|
|
|
|
|
Process *process = exe_ctx.process;
|
|
|
|
|
|
|
|
if (!process)
|
|
|
|
{
|
|
|
|
error_stream.PutCString ("error: invalid process\n");
|
|
|
|
return false;
|
|
|
|
}
|
2011-02-16 05:59:32 +08:00
|
|
|
|
2010-08-28 07:31:21 +08:00
|
|
|
//////////////////////////
|
|
|
|
// Parse the expression
|
|
|
|
//
|
2010-09-14 05:34:21 +08:00
|
|
|
|
2011-01-13 16:53:35 +08:00
|
|
|
bool keep_result_in_memory = false;
|
|
|
|
|
|
|
|
m_expr_decl_map.reset(new ClangExpressionDeclMap(keep_result_in_memory));
|
2010-12-03 09:38:59 +08:00
|
|
|
|
2011-05-24 05:40:23 +08:00
|
|
|
m_data_allocator.reset(new ProcessDataAllocator(*exe_ctx.process));
|
|
|
|
|
2011-08-02 02:18:33 +08:00
|
|
|
if (!m_expr_decl_map->WillParse(exe_ctx))
|
|
|
|
{
|
|
|
|
error_stream.PutCString ("error: current process state is unsuitable for expression parsing\n");
|
|
|
|
return false;
|
|
|
|
}
|
2010-08-28 07:31:21 +08:00
|
|
|
|
2011-02-16 05:59:32 +08:00
|
|
|
ClangExpressionParser parser(exe_ctx.GetBestExecutionContextScope(), *this);
|
2010-08-28 07:31:21 +08:00
|
|
|
|
|
|
|
unsigned num_errors = parser.Parse (error_stream);
|
|
|
|
|
|
|
|
if (num_errors)
|
|
|
|
{
|
|
|
|
error_stream.Printf ("error: %d errors parsing expression\n", num_errors);
|
2010-09-14 05:34:21 +08:00
|
|
|
|
|
|
|
m_expr_decl_map.reset();
|
|
|
|
|
2010-08-28 07:31:21 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
//////////////////////////////////
|
|
|
|
// JIT the output of the parser
|
|
|
|
//
|
2011-05-07 09:06:41 +08:00
|
|
|
|
|
|
|
lldb::ClangExpressionVariableSP const_result;
|
2011-05-24 05:40:23 +08:00
|
|
|
|
|
|
|
|
|
|
|
Error jit_error = parser.MakeJIT (m_jit_alloc, m_jit_start_addr, m_jit_end_addr, exe_ctx, m_data_allocator.get(), const_result);
|
|
|
|
|
|
|
|
if (log)
|
|
|
|
{
|
|
|
|
StreamString dump_string;
|
|
|
|
m_data_allocator->Dump(dump_string);
|
2010-08-28 07:31:21 +08:00
|
|
|
|
2011-05-24 05:40:23 +08:00
|
|
|
log->Printf("Data buffer contents:\n%s", dump_string.GetString().c_str());
|
|
|
|
}
|
2011-01-20 07:00:49 +08:00
|
|
|
|
|
|
|
if (exe_ctx.process && m_jit_start_addr != LLDB_INVALID_ADDRESS)
|
|
|
|
m_jit_process_sp = exe_ctx.process->GetSP();
|
2010-08-28 07:31:21 +08:00
|
|
|
|
2011-01-19 03:36:39 +08:00
|
|
|
#if 0
|
|
|
|
// jingham: look here
|
|
|
|
StreamFile logfile ("/tmp/exprs.txt", "a");
|
|
|
|
logfile.Printf ("0x%16.16llx: func = %s, source =\n%s\n",
|
2011-01-20 07:00:49 +08:00
|
|
|
m_jit_start_addr,
|
2011-01-19 03:36:39 +08:00
|
|
|
m_function_name.c_str(),
|
|
|
|
m_function_text.c_str());
|
|
|
|
#endif
|
|
|
|
|
2010-12-03 09:38:59 +08:00
|
|
|
m_expr_decl_map->DidParse();
|
|
|
|
|
2010-09-14 05:34:21 +08:00
|
|
|
m_expr_decl_map.reset();
|
|
|
|
|
2010-08-28 07:31:21 +08:00
|
|
|
if (jit_error.Success())
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2011-05-17 11:51:29 +08:00
|
|
|
const char *error_cstr = jit_error.AsCString();
|
|
|
|
if (error_cstr && error_cstr[0])
|
|
|
|
error_stream.Printf ("error: %s\n", error_cstr);
|
|
|
|
else
|
|
|
|
error_stream.Printf ("error: expression can't be interpreted or run\n", num_errors);
|
2010-08-28 07:31:21 +08:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|