2010-06-09 00:52:24 +08:00
|
|
|
//===-- ABISysV_x86_64.cpp --------------------------------------*- C++ -*-===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "ABISysV_x86_64.h"
|
|
|
|
|
|
|
|
#include "lldb/Core/ConstString.h"
|
|
|
|
#include "lldb/Core/DataExtractor.h"
|
|
|
|
#include "lldb/Core/Error.h"
|
2010-10-08 09:58:41 +08:00
|
|
|
#include "lldb/Core/Log.h"
|
2010-06-09 00:52:24 +08:00
|
|
|
#include "lldb/Core/Module.h"
|
|
|
|
#include "lldb/Core/PluginManager.h"
|
Cleaned up the ABI::PrepareTrivialCall() function to take three argument
pointers:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t *arg1_ptr,
lldb::addr_t *arg2_ptr,
lldb::addr_t *arg3_ptr) const = 0;
Prior to this it was:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t arg,
lldb::addr_t *this_arg,
lldb::addr_t *cmd_arg) const = 0;
This was because the function that called this slowly added more features to
be able to call a C++ member function that might have a "this" pointer, and
then later added "self + cmd" support for objective C. Cleaning this code up
and the code that calls it makes it easier to implement the functions for
new targets.
The MacOSX_arm::PrepareTrivialCall() is now filled in and ready for testing.
llvm-svn: 131221
2011-05-12 10:14:56 +08:00
|
|
|
#include "lldb/Core/RegisterValue.h"
|
2010-06-09 00:52:24 +08:00
|
|
|
#include "lldb/Core/Value.h"
|
2011-12-23 03:12:40 +08:00
|
|
|
#include "lldb/Core/ValueObjectConstResult.h"
|
|
|
|
#include "lldb/Core/ValueObjectRegister.h"
|
|
|
|
#include "lldb/Core/ValueObjectMemory.h"
|
2010-06-09 00:52:24 +08:00
|
|
|
#include "lldb/Symbol/ClangASTContext.h"
|
2011-05-12 02:39:18 +08:00
|
|
|
#include "lldb/Symbol/UnwindPlan.h"
|
2010-06-09 00:52:24 +08:00
|
|
|
#include "lldb/Target/Target.h"
|
|
|
|
#include "lldb/Target/Process.h"
|
|
|
|
#include "lldb/Target/RegisterContext.h"
|
|
|
|
#include "lldb/Target/StackFrame.h"
|
|
|
|
#include "lldb/Target/Thread.h"
|
|
|
|
|
|
|
|
#include "llvm/ADT/Triple.h"
|
|
|
|
|
|
|
|
using namespace lldb;
|
|
|
|
using namespace lldb_private;
|
|
|
|
|
|
|
|
static const char *pluginName = "ABISysV_x86_64";
|
|
|
|
static const char *pluginDesc = "System V ABI for x86_64 targets";
|
|
|
|
static const char *pluginShort = "abi.sysv-x86_64";
|
|
|
|
|
2011-05-12 02:39:18 +08:00
|
|
|
|
2011-08-22 10:49:39 +08:00
|
|
|
enum gcc_dwarf_regnums
|
|
|
|
{
|
|
|
|
gcc_dwarf_rax = 0,
|
|
|
|
gcc_dwarf_rdx,
|
|
|
|
gcc_dwarf_rcx,
|
|
|
|
gcc_dwarf_rbx,
|
|
|
|
gcc_dwarf_rsi,
|
|
|
|
gcc_dwarf_rdi,
|
|
|
|
gcc_dwarf_rbp,
|
|
|
|
gcc_dwarf_rsp,
|
|
|
|
gcc_dwarf_r8,
|
|
|
|
gcc_dwarf_r9,
|
|
|
|
gcc_dwarf_r10,
|
|
|
|
gcc_dwarf_r11,
|
|
|
|
gcc_dwarf_r12,
|
|
|
|
gcc_dwarf_r13,
|
|
|
|
gcc_dwarf_r14,
|
|
|
|
gcc_dwarf_r15,
|
|
|
|
gcc_dwarf_rip,
|
|
|
|
gcc_dwarf_xmm0,
|
|
|
|
gcc_dwarf_xmm1,
|
|
|
|
gcc_dwarf_xmm2,
|
|
|
|
gcc_dwarf_xmm3,
|
|
|
|
gcc_dwarf_xmm4,
|
|
|
|
gcc_dwarf_xmm5,
|
|
|
|
gcc_dwarf_xmm6,
|
|
|
|
gcc_dwarf_xmm7,
|
|
|
|
gcc_dwarf_xmm8,
|
|
|
|
gcc_dwarf_xmm9,
|
|
|
|
gcc_dwarf_xmm10,
|
|
|
|
gcc_dwarf_xmm11,
|
|
|
|
gcc_dwarf_xmm12,
|
|
|
|
gcc_dwarf_xmm13,
|
|
|
|
gcc_dwarf_xmm14,
|
|
|
|
gcc_dwarf_xmm15,
|
|
|
|
gcc_dwarf_stmm0,
|
|
|
|
gcc_dwarf_stmm1,
|
|
|
|
gcc_dwarf_stmm2,
|
|
|
|
gcc_dwarf_stmm3,
|
|
|
|
gcc_dwarf_stmm4,
|
|
|
|
gcc_dwarf_stmm5,
|
|
|
|
gcc_dwarf_stmm6,
|
|
|
|
gcc_dwarf_stmm7,
|
|
|
|
gcc_dwarf_ymm0 = gcc_dwarf_xmm0,
|
|
|
|
gcc_dwarf_ymm1 = gcc_dwarf_xmm1,
|
|
|
|
gcc_dwarf_ymm2 = gcc_dwarf_xmm2,
|
|
|
|
gcc_dwarf_ymm3 = gcc_dwarf_xmm3,
|
|
|
|
gcc_dwarf_ymm4 = gcc_dwarf_xmm4,
|
|
|
|
gcc_dwarf_ymm5 = gcc_dwarf_xmm5,
|
|
|
|
gcc_dwarf_ymm6 = gcc_dwarf_xmm6,
|
|
|
|
gcc_dwarf_ymm7 = gcc_dwarf_xmm7,
|
|
|
|
gcc_dwarf_ymm8 = gcc_dwarf_xmm8,
|
|
|
|
gcc_dwarf_ymm9 = gcc_dwarf_xmm9,
|
|
|
|
gcc_dwarf_ymm10 = gcc_dwarf_xmm10,
|
|
|
|
gcc_dwarf_ymm11 = gcc_dwarf_xmm11,
|
|
|
|
gcc_dwarf_ymm12 = gcc_dwarf_xmm12,
|
|
|
|
gcc_dwarf_ymm13 = gcc_dwarf_xmm13,
|
|
|
|
gcc_dwarf_ymm14 = gcc_dwarf_xmm14,
|
|
|
|
gcc_dwarf_ymm15 = gcc_dwarf_xmm15
|
|
|
|
};
|
|
|
|
|
|
|
|
enum gdb_regnums
|
|
|
|
{
|
|
|
|
gdb_rax = 0,
|
|
|
|
gdb_rbx = 1,
|
|
|
|
gdb_rcx = 2,
|
|
|
|
gdb_rdx = 3,
|
|
|
|
gdb_rsi = 4,
|
|
|
|
gdb_rdi = 5,
|
|
|
|
gdb_rbp = 6,
|
|
|
|
gdb_rsp = 7,
|
|
|
|
gdb_r8 = 8,
|
|
|
|
gdb_r9 = 9,
|
|
|
|
gdb_r10 = 10,
|
|
|
|
gdb_r11 = 11,
|
|
|
|
gdb_r12 = 12,
|
|
|
|
gdb_r13 = 13,
|
|
|
|
gdb_r14 = 14,
|
|
|
|
gdb_r15 = 15,
|
|
|
|
gdb_rip = 16,
|
|
|
|
gdb_rflags = 17,
|
|
|
|
gdb_cs = 18,
|
|
|
|
gdb_ss = 19,
|
|
|
|
gdb_ds = 20,
|
|
|
|
gdb_es = 21,
|
|
|
|
gdb_fs = 22,
|
|
|
|
gdb_gs = 23,
|
|
|
|
gdb_stmm0 = 24,
|
|
|
|
gdb_stmm1 = 25,
|
|
|
|
gdb_stmm2 = 26,
|
|
|
|
gdb_stmm3 = 27,
|
|
|
|
gdb_stmm4 = 28,
|
|
|
|
gdb_stmm5 = 29,
|
|
|
|
gdb_stmm6 = 30,
|
|
|
|
gdb_stmm7 = 31,
|
|
|
|
gdb_fctrl = 32, gdb_fcw = gdb_fctrl,
|
|
|
|
gdb_fstat = 33, gdb_fsw = gdb_fstat,
|
|
|
|
gdb_ftag = 34, gdb_ftw = gdb_ftag,
|
|
|
|
gdb_fiseg = 35, gdb_fpu_cs = gdb_fiseg,
|
|
|
|
gdb_fioff = 36, gdb_ip = gdb_fioff,
|
|
|
|
gdb_foseg = 37, gdb_fpu_ds = gdb_foseg,
|
|
|
|
gdb_fooff = 38, gdb_dp = gdb_fooff,
|
|
|
|
gdb_fop = 39,
|
|
|
|
gdb_xmm0 = 40,
|
|
|
|
gdb_xmm1 = 41,
|
|
|
|
gdb_xmm2 = 42,
|
|
|
|
gdb_xmm3 = 43,
|
|
|
|
gdb_xmm4 = 44,
|
|
|
|
gdb_xmm5 = 45,
|
|
|
|
gdb_xmm6 = 46,
|
|
|
|
gdb_xmm7 = 47,
|
|
|
|
gdb_xmm8 = 48,
|
|
|
|
gdb_xmm9 = 49,
|
|
|
|
gdb_xmm10 = 50,
|
|
|
|
gdb_xmm11 = 51,
|
|
|
|
gdb_xmm12 = 52,
|
|
|
|
gdb_xmm13 = 53,
|
|
|
|
gdb_xmm14 = 54,
|
|
|
|
gdb_xmm15 = 55,
|
|
|
|
gdb_mxcsr = 56,
|
|
|
|
gdb_ymm0 = gdb_xmm0,
|
|
|
|
gdb_ymm1 = gdb_xmm1,
|
|
|
|
gdb_ymm2 = gdb_xmm2,
|
|
|
|
gdb_ymm3 = gdb_xmm3,
|
|
|
|
gdb_ymm4 = gdb_xmm4,
|
|
|
|
gdb_ymm5 = gdb_xmm5,
|
|
|
|
gdb_ymm6 = gdb_xmm6,
|
|
|
|
gdb_ymm7 = gdb_xmm7,
|
|
|
|
gdb_ymm8 = gdb_xmm8,
|
|
|
|
gdb_ymm9 = gdb_xmm9,
|
|
|
|
gdb_ymm10 = gdb_xmm10,
|
|
|
|
gdb_ymm11 = gdb_xmm11,
|
|
|
|
gdb_ymm12 = gdb_xmm12,
|
|
|
|
gdb_ymm13 = gdb_xmm13,
|
|
|
|
gdb_ymm14 = gdb_xmm14,
|
|
|
|
gdb_ymm15 = gdb_xmm15
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
static RegisterInfo g_register_infos[] =
|
|
|
|
{
|
2012-02-29 09:07:59 +08:00
|
|
|
// NAME ALT SZ OFF ENCODING FORMAT COMPILER DWARF GENERIC GDB LLDB NATIVE VALUE REGS INVALIDATE REGS
|
|
|
|
// ======== ======= == === ============= =================== ======================= ===================== =========================== ===================== ====================== ========== ===============
|
|
|
|
{ "rax" , NULL, 8, 0, eEncodingUint , eFormatHex , { gcc_dwarf_rax , gcc_dwarf_rax , LLDB_INVALID_REGNUM , gdb_rax , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "rbx" , NULL, 8, 0, eEncodingUint , eFormatHex , { gcc_dwarf_rbx , gcc_dwarf_rbx , LLDB_INVALID_REGNUM , gdb_rbx , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "rcx" , "arg4", 8, 0, eEncodingUint , eFormatHex , { gcc_dwarf_rcx , gcc_dwarf_rcx , LLDB_REGNUM_GENERIC_ARG4 , gdb_rcx , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "rdx" , "arg3", 8, 0, eEncodingUint , eFormatHex , { gcc_dwarf_rdx , gcc_dwarf_rdx , LLDB_REGNUM_GENERIC_ARG3 , gdb_rdx , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "rsi" , "arg2", 8, 0, eEncodingUint , eFormatHex , { gcc_dwarf_rsi , gcc_dwarf_rsi , LLDB_REGNUM_GENERIC_ARG2 , gdb_rsi , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "rdi" , "arg1", 8, 0, eEncodingUint , eFormatHex , { gcc_dwarf_rdi , gcc_dwarf_rdi , LLDB_REGNUM_GENERIC_ARG1 , gdb_rdi , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "rbp" , "fp", 8, 0, eEncodingUint , eFormatHex , { gcc_dwarf_rbp , gcc_dwarf_rbp , LLDB_REGNUM_GENERIC_FP , gdb_rbp , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "rsp" , "sp", 8, 0, eEncodingUint , eFormatHex , { gcc_dwarf_rsp , gcc_dwarf_rsp , LLDB_REGNUM_GENERIC_SP , gdb_rsp , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "r8" , "arg5", 8, 0, eEncodingUint , eFormatHex , { gcc_dwarf_r8 , gcc_dwarf_r8 , LLDB_REGNUM_GENERIC_ARG5 , gdb_r8 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "r9" , "arg6", 8, 0, eEncodingUint , eFormatHex , { gcc_dwarf_r9 , gcc_dwarf_r9 , LLDB_REGNUM_GENERIC_ARG6 , gdb_r9 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "r10" , NULL, 8, 0, eEncodingUint , eFormatHex , { gcc_dwarf_r10 , gcc_dwarf_r10 , LLDB_INVALID_REGNUM , gdb_r10 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "r11" , NULL, 8, 0, eEncodingUint , eFormatHex , { gcc_dwarf_r11 , gcc_dwarf_r11 , LLDB_INVALID_REGNUM , gdb_r11 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "r12" , NULL, 8, 0, eEncodingUint , eFormatHex , { gcc_dwarf_r12 , gcc_dwarf_r12 , LLDB_INVALID_REGNUM , gdb_r12 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "r13" , NULL, 8, 0, eEncodingUint , eFormatHex , { gcc_dwarf_r13 , gcc_dwarf_r13 , LLDB_INVALID_REGNUM , gdb_r13 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "r14" , NULL, 8, 0, eEncodingUint , eFormatHex , { gcc_dwarf_r14 , gcc_dwarf_r14 , LLDB_INVALID_REGNUM , gdb_r14 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "r15" , NULL, 8, 0, eEncodingUint , eFormatHex , { gcc_dwarf_r15 , gcc_dwarf_r15 , LLDB_INVALID_REGNUM , gdb_r15 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "rip" , "pc", 8, 0, eEncodingUint , eFormatHex , { gcc_dwarf_rip , gcc_dwarf_rip , LLDB_REGNUM_GENERIC_PC , gdb_rip , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "rflags", NULL, 4, 0, eEncodingUint , eFormatHex , { LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , LLDB_REGNUM_GENERIC_FLAGS , gdb_rflags , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "cs" , NULL, 4, 0, eEncodingUint , eFormatHex , { LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , gdb_cs , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "ss" , NULL, 4, 0, eEncodingUint , eFormatHex , { LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , gdb_ss , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "ds" , NULL, 4, 0, eEncodingUint , eFormatHex , { LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , gdb_ds , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "es" , NULL, 4, 0, eEncodingUint , eFormatHex , { LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , gdb_es , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "fs" , NULL, 4, 0, eEncodingUint , eFormatHex , { LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , gdb_fs , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "gs" , NULL, 4, 0, eEncodingUint , eFormatHex , { LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , gdb_gs , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "stmm0" , NULL, 10, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_stmm0 , gcc_dwarf_stmm0 , LLDB_INVALID_REGNUM , gdb_stmm0 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "stmm1" , NULL, 10, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_stmm1 , gcc_dwarf_stmm1 , LLDB_INVALID_REGNUM , gdb_stmm1 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "stmm2" , NULL, 10, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_stmm2 , gcc_dwarf_stmm2 , LLDB_INVALID_REGNUM , gdb_stmm2 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "stmm3" , NULL, 10, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_stmm3 , gcc_dwarf_stmm3 , LLDB_INVALID_REGNUM , gdb_stmm3 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "stmm4" , NULL, 10, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_stmm4 , gcc_dwarf_stmm4 , LLDB_INVALID_REGNUM , gdb_stmm4 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "stmm5" , NULL, 10, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_stmm5 , gcc_dwarf_stmm5 , LLDB_INVALID_REGNUM , gdb_stmm5 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "stmm6" , NULL, 10, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_stmm6 , gcc_dwarf_stmm6 , LLDB_INVALID_REGNUM , gdb_stmm6 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "stmm7" , NULL, 10, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_stmm7 , gcc_dwarf_stmm7 , LLDB_INVALID_REGNUM , gdb_stmm7 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "fctrl" , NULL, 4, 0, eEncodingUint , eFormatHex , { LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , gdb_fctrl , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "fstat" , NULL, 4, 0, eEncodingUint , eFormatHex , { LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , gdb_fstat , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "ftag" , NULL, 4, 0, eEncodingUint , eFormatHex , { LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , gdb_ftag , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "fiseg" , NULL, 4, 0, eEncodingUint , eFormatHex , { LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , gdb_fiseg , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "fioff" , NULL, 4, 0, eEncodingUint , eFormatHex , { LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , gdb_fioff , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "foseg" , NULL, 4, 0, eEncodingUint , eFormatHex , { LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , gdb_foseg , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "fooff" , NULL, 4, 0, eEncodingUint , eFormatHex , { LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , gdb_fooff , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "fop" , NULL, 4, 0, eEncodingUint , eFormatHex , { LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , gdb_fop , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "xmm0" , NULL, 16, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_xmm0 , gcc_dwarf_xmm0 , LLDB_INVALID_REGNUM , gdb_xmm0 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "xmm1" , NULL, 16, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_xmm1 , gcc_dwarf_xmm1 , LLDB_INVALID_REGNUM , gdb_xmm1 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "xmm2" , NULL, 16, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_xmm2 , gcc_dwarf_xmm2 , LLDB_INVALID_REGNUM , gdb_xmm2 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "xmm3" , NULL, 16, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_xmm3 , gcc_dwarf_xmm3 , LLDB_INVALID_REGNUM , gdb_xmm3 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "xmm4" , NULL, 16, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_xmm4 , gcc_dwarf_xmm4 , LLDB_INVALID_REGNUM , gdb_xmm4 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "xmm5" , NULL, 16, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_xmm5 , gcc_dwarf_xmm5 , LLDB_INVALID_REGNUM , gdb_xmm5 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "xmm6" , NULL, 16, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_xmm6 , gcc_dwarf_xmm6 , LLDB_INVALID_REGNUM , gdb_xmm6 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "xmm7" , NULL, 16, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_xmm7 , gcc_dwarf_xmm7 , LLDB_INVALID_REGNUM , gdb_xmm7 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "xmm8" , NULL, 16, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_xmm8 , gcc_dwarf_xmm8 , LLDB_INVALID_REGNUM , gdb_xmm8 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "xmm9" , NULL, 16, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_xmm9 , gcc_dwarf_xmm9 , LLDB_INVALID_REGNUM , gdb_xmm9 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "xmm10" , NULL, 16, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_xmm10 , gcc_dwarf_xmm10 , LLDB_INVALID_REGNUM , gdb_xmm10 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "xmm11" , NULL, 16, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_xmm11 , gcc_dwarf_xmm11 , LLDB_INVALID_REGNUM , gdb_xmm11 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "xmm12" , NULL, 16, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_xmm12 , gcc_dwarf_xmm12 , LLDB_INVALID_REGNUM , gdb_xmm12 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "xmm13" , NULL, 16, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_xmm13 , gcc_dwarf_xmm13 , LLDB_INVALID_REGNUM , gdb_xmm13 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "xmm14" , NULL, 16, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_xmm14 , gcc_dwarf_xmm14 , LLDB_INVALID_REGNUM , gdb_xmm14 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "xmm15" , NULL, 16, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_xmm15 , gcc_dwarf_xmm15 , LLDB_INVALID_REGNUM , gdb_xmm15 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "mxcsr" , NULL, 4, 0, eEncodingUint , eFormatHex , { LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , LLDB_INVALID_REGNUM , gdb_mxcsr , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "ymm0" , NULL, 32, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_ymm0 , gcc_dwarf_ymm0 , LLDB_INVALID_REGNUM , gdb_ymm0 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "ymm1" , NULL, 32, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_ymm1 , gcc_dwarf_ymm1 , LLDB_INVALID_REGNUM , gdb_ymm1 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "ymm2" , NULL, 32, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_ymm2 , gcc_dwarf_ymm2 , LLDB_INVALID_REGNUM , gdb_ymm2 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "ymm3" , NULL, 32, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_ymm3 , gcc_dwarf_ymm3 , LLDB_INVALID_REGNUM , gdb_ymm3 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "ymm4" , NULL, 32, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_ymm4 , gcc_dwarf_ymm4 , LLDB_INVALID_REGNUM , gdb_ymm4 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "ymm5" , NULL, 32, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_ymm5 , gcc_dwarf_ymm5 , LLDB_INVALID_REGNUM , gdb_ymm5 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "ymm6" , NULL, 32, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_ymm6 , gcc_dwarf_ymm6 , LLDB_INVALID_REGNUM , gdb_ymm6 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "ymm7" , NULL, 32, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_ymm7 , gcc_dwarf_ymm7 , LLDB_INVALID_REGNUM , gdb_ymm7 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "ymm8" , NULL, 32, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_ymm8 , gcc_dwarf_ymm8 , LLDB_INVALID_REGNUM , gdb_ymm8 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "ymm9" , NULL, 32, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_ymm9 , gcc_dwarf_ymm9 , LLDB_INVALID_REGNUM , gdb_ymm9 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "ymm10" , NULL, 32, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_ymm10 , gcc_dwarf_ymm10 , LLDB_INVALID_REGNUM , gdb_ymm10 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "ymm11" , NULL, 32, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_ymm11 , gcc_dwarf_ymm11 , LLDB_INVALID_REGNUM , gdb_ymm11 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "ymm12" , NULL, 32, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_ymm12 , gcc_dwarf_ymm12 , LLDB_INVALID_REGNUM , gdb_ymm12 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "ymm13" , NULL, 32, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_ymm13 , gcc_dwarf_ymm13 , LLDB_INVALID_REGNUM , gdb_ymm13 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "ymm14" , NULL, 32, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_ymm14 , gcc_dwarf_ymm14 , LLDB_INVALID_REGNUM , gdb_ymm14 , LLDB_INVALID_REGNUM }, NULL, NULL},
|
|
|
|
{ "ymm15" , NULL, 32, 0, eEncodingVector, eFormatVectorOfUInt8, { gcc_dwarf_ymm15 , gcc_dwarf_ymm15 , LLDB_INVALID_REGNUM , gdb_ymm15 , LLDB_INVALID_REGNUM }, NULL, NULL}
|
2011-08-22 10:49:39 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
static const uint32_t k_num_register_infos = sizeof(g_register_infos)/sizeof(RegisterInfo);
|
|
|
|
static bool g_register_info_names_constified = false;
|
|
|
|
|
|
|
|
const lldb_private::RegisterInfo *
|
|
|
|
ABISysV_x86_64::GetRegisterInfoArray (uint32_t &count)
|
|
|
|
{
|
|
|
|
// Make the C-string names and alt_names for the register infos into const
|
|
|
|
// C-string values by having the ConstString unique the names in the global
|
|
|
|
// constant C-string pool.
|
|
|
|
if (!g_register_info_names_constified)
|
|
|
|
{
|
|
|
|
g_register_info_names_constified = true;
|
|
|
|
for (uint32_t i=0; i<k_num_register_infos; ++i)
|
|
|
|
{
|
|
|
|
if (g_register_infos[i].name)
|
|
|
|
g_register_infos[i].name = ConstString(g_register_infos[i].name).GetCString();
|
|
|
|
if (g_register_infos[i].alt_name)
|
|
|
|
g_register_infos[i].alt_name = ConstString(g_register_infos[i].alt_name).GetCString();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
count = k_num_register_infos;
|
|
|
|
return g_register_infos;
|
|
|
|
}
|
|
|
|
|
2011-05-12 02:39:18 +08:00
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
size_t
|
|
|
|
ABISysV_x86_64::GetRedZoneSize () const
|
|
|
|
{
|
|
|
|
return 128;
|
|
|
|
}
|
|
|
|
|
|
|
|
//------------------------------------------------------------------
|
|
|
|
// Static Functions
|
|
|
|
//------------------------------------------------------------------
|
2011-05-12 02:39:18 +08:00
|
|
|
ABISP
|
2011-02-16 05:59:32 +08:00
|
|
|
ABISysV_x86_64::CreateInstance (const ArchSpec &arch)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2011-05-12 02:39:18 +08:00
|
|
|
static ABISP g_abi_sp;
|
2011-02-16 05:59:32 +08:00
|
|
|
if (arch.GetTriple().getArch() == llvm::Triple::x86_64)
|
2011-05-12 02:39:18 +08:00
|
|
|
{
|
|
|
|
if (!g_abi_sp)
|
|
|
|
g_abi_sp.reset (new ABISysV_x86_64);
|
|
|
|
return g_abi_sp;
|
|
|
|
}
|
|
|
|
return ABISP();
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
ABISysV_x86_64::PrepareTrivialCall (Thread &thread,
|
2011-05-14 09:50:35 +08:00
|
|
|
addr_t sp,
|
|
|
|
addr_t func_addr,
|
|
|
|
addr_t return_addr,
|
|
|
|
addr_t *arg1_ptr,
|
|
|
|
addr_t *arg2_ptr,
|
|
|
|
addr_t *arg3_ptr,
|
|
|
|
addr_t *arg4_ptr,
|
|
|
|
addr_t *arg5_ptr,
|
|
|
|
addr_t *arg6_ptr) const
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2010-11-06 09:53:30 +08:00
|
|
|
LogSP log(lldb_private::GetLogIfAllCategoriesSet (LIBLLDB_LOG_EXPRESSIONS));
|
2010-10-08 09:58:41 +08:00
|
|
|
|
|
|
|
if (log)
|
Cleaned up the ABI::PrepareTrivialCall() function to take three argument
pointers:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t *arg1_ptr,
lldb::addr_t *arg2_ptr,
lldb::addr_t *arg3_ptr) const = 0;
Prior to this it was:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t arg,
lldb::addr_t *this_arg,
lldb::addr_t *cmd_arg) const = 0;
This was because the function that called this slowly added more features to
be able to call a C++ member function that might have a "this" pointer, and
then later added "self + cmd" support for objective C. Cleaning this code up
and the code that calls it makes it easier to implement the functions for
new targets.
The MacOSX_arm::PrepareTrivialCall() is now filled in and ready for testing.
llvm-svn: 131221
2011-05-12 10:14:56 +08:00
|
|
|
log->Printf("ABISysV_x86_64::PrepareTrivialCall\n(\n thread = %p\n sp = 0x%llx\n func_addr = 0x%llx\n return_addr = 0x%llx\n arg1_ptr = %p (0x%llx)\n arg2_ptr = %p (0x%llx)\n arg3_ptr = %p (0x%llx)\n)",
|
2010-10-08 09:58:41 +08:00
|
|
|
(void*)&thread,
|
|
|
|
(uint64_t)sp,
|
Cleaned up the ABI::PrepareTrivialCall() function to take three argument
pointers:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t *arg1_ptr,
lldb::addr_t *arg2_ptr,
lldb::addr_t *arg3_ptr) const = 0;
Prior to this it was:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t arg,
lldb::addr_t *this_arg,
lldb::addr_t *cmd_arg) const = 0;
This was because the function that called this slowly added more features to
be able to call a C++ member function that might have a "this" pointer, and
then later added "self + cmd" support for objective C. Cleaning this code up
and the code that calls it makes it easier to implement the functions for
new targets.
The MacOSX_arm::PrepareTrivialCall() is now filled in and ready for testing.
llvm-svn: 131221
2011-05-12 10:14:56 +08:00
|
|
|
(uint64_t)func_addr,
|
|
|
|
(uint64_t)return_addr,
|
|
|
|
arg1_ptr, arg1_ptr ? (uint64_t)*arg1_ptr : (uint64_t) 0,
|
|
|
|
arg2_ptr, arg2_ptr ? (uint64_t)*arg2_ptr : (uint64_t) 0,
|
|
|
|
arg3_ptr, arg3_ptr ? (uint64_t)*arg3_ptr : (uint64_t) 0);
|
2010-10-08 09:58:41 +08:00
|
|
|
|
2011-01-07 06:15:06 +08:00
|
|
|
RegisterContext *reg_ctx = thread.GetRegisterContext().get();
|
2010-06-09 00:52:24 +08:00
|
|
|
if (!reg_ctx)
|
|
|
|
return false;
|
2010-09-08 05:23:34 +08:00
|
|
|
|
2011-05-15 09:25:55 +08:00
|
|
|
const RegisterInfo *reg_info = NULL;
|
Cleaned up the ABI::PrepareTrivialCall() function to take three argument
pointers:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t *arg1_ptr,
lldb::addr_t *arg2_ptr,
lldb::addr_t *arg3_ptr) const = 0;
Prior to this it was:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t arg,
lldb::addr_t *this_arg,
lldb::addr_t *cmd_arg) const = 0;
This was because the function that called this slowly added more features to
be able to call a C++ member function that might have a "this" pointer, and
then later added "self + cmd" support for objective C. Cleaning this code up
and the code that calls it makes it easier to implement the functions for
new targets.
The MacOSX_arm::PrepareTrivialCall() is now filled in and ready for testing.
llvm-svn: 131221
2011-05-12 10:14:56 +08:00
|
|
|
if (arg1_ptr)
|
Removed the hacky "#define this ___clang_this" handler
for C++ classes. Replaced it with a less hacky approach:
- If an expression is defined in the context of a
method of class A, then that expression is wrapped as
___clang_class::___clang_expr(void*) { ... }
instead of ___clang_expr(void*) { ... }.
- ___clang_class is resolved as the type of the target
of the "this" pointer in the method the expression
is defined in.
- When reporting the type of ___clang_class, a method
with the signature ___clang_expr(void*) is added to
that class, so that Clang doesn't complain about a
method being defined without a corresponding
declaration.
- Whenever the expression gets called, "this" gets
looked up, type-checked, and then passed in as the
first argument.
This required the following changes:
- The ABIs were changed to support passing of the "this"
pointer as part of trivial calls.
- ThreadPlanCallFunction and ClangFunction were changed
to support passing of an optional "this" pointer.
- ClangUserExpression was extended to perform the
wrapping described above.
- ClangASTSource was changed to revert the changes
required by the hack.
- ClangExpressionParser, IRForTarget, and
ClangExpressionDeclMap were changed to handle
different manglings of ___clang_expr flexibly. This
meant no longer searching for a function called
___clang_expr, but rather looking for a function whose
name *contains* ___clang_expr.
- ClangExpressionParser and ClangExpressionDeclMap now
remember whether "this" is required, and know how to
look it up as necessary.
A few inheritance bugs remain, and I'm trying to resolve
these. But it is now possible to use "this" as well as
refer implicitly to member variables, when in the proper
context.
llvm-svn: 114384
2010-09-21 08:44:12 +08:00
|
|
|
{
|
2011-05-15 09:25:55 +08:00
|
|
|
reg_info = reg_ctx->GetRegisterInfoByName("rdi", 0);
|
2010-10-08 09:58:41 +08:00
|
|
|
if (log)
|
2011-05-15 09:25:55 +08:00
|
|
|
log->Printf("About to write arg1 (0x%llx) into %s", (uint64_t)*arg1_ptr, reg_info->name);
|
Cleaned up the ABI::PrepareTrivialCall() function to take three argument
pointers:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t *arg1_ptr,
lldb::addr_t *arg2_ptr,
lldb::addr_t *arg3_ptr) const = 0;
Prior to this it was:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t arg,
lldb::addr_t *this_arg,
lldb::addr_t *cmd_arg) const = 0;
This was because the function that called this slowly added more features to
be able to call a C++ member function that might have a "this" pointer, and
then later added "self + cmd" support for objective C. Cleaning this code up
and the code that calls it makes it easier to implement the functions for
new targets.
The MacOSX_arm::PrepareTrivialCall() is now filled in and ready for testing.
llvm-svn: 131221
2011-05-12 10:14:56 +08:00
|
|
|
|
2011-05-15 09:25:55 +08:00
|
|
|
if (!reg_ctx->WriteRegisterFromUnsigned (reg_info, *arg1_ptr))
|
Removed the hacky "#define this ___clang_this" handler
for C++ classes. Replaced it with a less hacky approach:
- If an expression is defined in the context of a
method of class A, then that expression is wrapped as
___clang_class::___clang_expr(void*) { ... }
instead of ___clang_expr(void*) { ... }.
- ___clang_class is resolved as the type of the target
of the "this" pointer in the method the expression
is defined in.
- When reporting the type of ___clang_class, a method
with the signature ___clang_expr(void*) is added to
that class, so that Clang doesn't complain about a
method being defined without a corresponding
declaration.
- Whenever the expression gets called, "this" gets
looked up, type-checked, and then passed in as the
first argument.
This required the following changes:
- The ABIs were changed to support passing of the "this"
pointer as part of trivial calls.
- ThreadPlanCallFunction and ClangFunction were changed
to support passing of an optional "this" pointer.
- ClangUserExpression was extended to perform the
wrapping described above.
- ClangASTSource was changed to revert the changes
required by the hack.
- ClangExpressionParser, IRForTarget, and
ClangExpressionDeclMap were changed to handle
different manglings of ___clang_expr flexibly. This
meant no longer searching for a function called
___clang_expr, but rather looking for a function whose
name *contains* ___clang_expr.
- ClangExpressionParser and ClangExpressionDeclMap now
remember whether "this" is required, and know how to
look it up as necessary.
A few inheritance bugs remain, and I'm trying to resolve
these. But it is now possible to use "this" as well as
refer implicitly to member variables, when in the proper
context.
llvm-svn: 114384
2010-09-21 08:44:12 +08:00
|
|
|
return false;
|
Cleaned up the ABI::PrepareTrivialCall() function to take three argument
pointers:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t *arg1_ptr,
lldb::addr_t *arg2_ptr,
lldb::addr_t *arg3_ptr) const = 0;
Prior to this it was:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t arg,
lldb::addr_t *this_arg,
lldb::addr_t *cmd_arg) const = 0;
This was because the function that called this slowly added more features to
be able to call a C++ member function that might have a "this" pointer, and
then later added "self + cmd" support for objective C. Cleaning this code up
and the code that calls it makes it easier to implement the functions for
new targets.
The MacOSX_arm::PrepareTrivialCall() is now filled in and ready for testing.
llvm-svn: 131221
2011-05-12 10:14:56 +08:00
|
|
|
|
|
|
|
if (arg2_ptr)
|
|
|
|
{
|
2011-05-15 09:25:55 +08:00
|
|
|
reg_info = reg_ctx->GetRegisterInfoByName("rsi", 0);
|
Cleaned up the ABI::PrepareTrivialCall() function to take three argument
pointers:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t *arg1_ptr,
lldb::addr_t *arg2_ptr,
lldb::addr_t *arg3_ptr) const = 0;
Prior to this it was:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t arg,
lldb::addr_t *this_arg,
lldb::addr_t *cmd_arg) const = 0;
This was because the function that called this slowly added more features to
be able to call a C++ member function that might have a "this" pointer, and
then later added "self + cmd" support for objective C. Cleaning this code up
and the code that calls it makes it easier to implement the functions for
new targets.
The MacOSX_arm::PrepareTrivialCall() is now filled in and ready for testing.
llvm-svn: 131221
2011-05-12 10:14:56 +08:00
|
|
|
if (log)
|
2011-05-15 09:25:55 +08:00
|
|
|
log->Printf("About to write arg2 (0x%llx) into %s", (uint64_t)*arg2_ptr, reg_info->name);
|
|
|
|
if (!reg_ctx->WriteRegisterFromUnsigned (reg_info, *arg2_ptr))
|
Cleaned up the ABI::PrepareTrivialCall() function to take three argument
pointers:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t *arg1_ptr,
lldb::addr_t *arg2_ptr,
lldb::addr_t *arg3_ptr) const = 0;
Prior to this it was:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t arg,
lldb::addr_t *this_arg,
lldb::addr_t *cmd_arg) const = 0;
This was because the function that called this slowly added more features to
be able to call a C++ member function that might have a "this" pointer, and
then later added "self + cmd" support for objective C. Cleaning this code up
and the code that calls it makes it easier to implement the functions for
new targets.
The MacOSX_arm::PrepareTrivialCall() is now filled in and ready for testing.
llvm-svn: 131221
2011-05-12 10:14:56 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
if (arg3_ptr)
|
|
|
|
{
|
2011-05-15 09:25:55 +08:00
|
|
|
reg_info = reg_ctx->GetRegisterInfoByName("rdx", 0);
|
Cleaned up the ABI::PrepareTrivialCall() function to take three argument
pointers:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t *arg1_ptr,
lldb::addr_t *arg2_ptr,
lldb::addr_t *arg3_ptr) const = 0;
Prior to this it was:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t arg,
lldb::addr_t *this_arg,
lldb::addr_t *cmd_arg) const = 0;
This was because the function that called this slowly added more features to
be able to call a C++ member function that might have a "this" pointer, and
then later added "self + cmd" support for objective C. Cleaning this code up
and the code that calls it makes it easier to implement the functions for
new targets.
The MacOSX_arm::PrepareTrivialCall() is now filled in and ready for testing.
llvm-svn: 131221
2011-05-12 10:14:56 +08:00
|
|
|
if (log)
|
2011-05-15 09:25:55 +08:00
|
|
|
log->Printf("About to write arg3 (0x%llx) into %s", (uint64_t)*arg3_ptr, reg_info->name);
|
|
|
|
if (!reg_ctx->WriteRegisterFromUnsigned (reg_info, *arg3_ptr))
|
Cleaned up the ABI::PrepareTrivialCall() function to take three argument
pointers:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t *arg1_ptr,
lldb::addr_t *arg2_ptr,
lldb::addr_t *arg3_ptr) const = 0;
Prior to this it was:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t arg,
lldb::addr_t *this_arg,
lldb::addr_t *cmd_arg) const = 0;
This was because the function that called this slowly added more features to
be able to call a C++ member function that might have a "this" pointer, and
then later added "self + cmd" support for objective C. Cleaning this code up
and the code that calls it makes it easier to implement the functions for
new targets.
The MacOSX_arm::PrepareTrivialCall() is now filled in and ready for testing.
llvm-svn: 131221
2011-05-12 10:14:56 +08:00
|
|
|
return false;
|
2011-05-15 09:25:55 +08:00
|
|
|
|
|
|
|
if (arg4_ptr)
|
|
|
|
{
|
|
|
|
reg_info = reg_ctx->GetRegisterInfoByName("rcx", 0);
|
|
|
|
if (log)
|
|
|
|
log->Printf("About to write arg4 (0x%llx) into %s", (uint64_t)*arg4_ptr, reg_info->name);
|
|
|
|
if (!reg_ctx->WriteRegisterFromUnsigned (reg_info, *arg4_ptr))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (arg5_ptr)
|
|
|
|
{
|
|
|
|
reg_info = reg_ctx->GetRegisterInfoByName("r8", 0);
|
|
|
|
if (log)
|
|
|
|
log->Printf("About to write arg5 (0x%llx) into %s", (uint64_t)*arg5_ptr, reg_info->name);
|
|
|
|
if (!reg_ctx->WriteRegisterFromUnsigned (reg_info, *arg5_ptr))
|
|
|
|
return false;
|
|
|
|
|
|
|
|
if (arg6_ptr)
|
|
|
|
{
|
|
|
|
reg_info = reg_ctx->GetRegisterInfoByName("r9", 0);
|
|
|
|
if (log)
|
|
|
|
log->Printf("About to write arg6 (0x%llx) into %s", (uint64_t)*arg6_ptr, reg_info->name);
|
|
|
|
if (!reg_ctx->WriteRegisterFromUnsigned (reg_info, *arg6_ptr))
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
Cleaned up the ABI::PrepareTrivialCall() function to take three argument
pointers:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t *arg1_ptr,
lldb::addr_t *arg2_ptr,
lldb::addr_t *arg3_ptr) const = 0;
Prior to this it was:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t arg,
lldb::addr_t *this_arg,
lldb::addr_t *cmd_arg) const = 0;
This was because the function that called this slowly added more features to
be able to call a C++ member function that might have a "this" pointer, and
then later added "self + cmd" support for objective C. Cleaning this code up
and the code that calls it makes it easier to implement the functions for
new targets.
The MacOSX_arm::PrepareTrivialCall() is now filled in and ready for testing.
llvm-svn: 131221
2011-05-12 10:14:56 +08:00
|
|
|
}
|
|
|
|
}
|
Removed the hacky "#define this ___clang_this" handler
for C++ classes. Replaced it with a less hacky approach:
- If an expression is defined in the context of a
method of class A, then that expression is wrapped as
___clang_class::___clang_expr(void*) { ... }
instead of ___clang_expr(void*) { ... }.
- ___clang_class is resolved as the type of the target
of the "this" pointer in the method the expression
is defined in.
- When reporting the type of ___clang_class, a method
with the signature ___clang_expr(void*) is added to
that class, so that Clang doesn't complain about a
method being defined without a corresponding
declaration.
- Whenever the expression gets called, "this" gets
looked up, type-checked, and then passed in as the
first argument.
This required the following changes:
- The ABIs were changed to support passing of the "this"
pointer as part of trivial calls.
- ThreadPlanCallFunction and ClangFunction were changed
to support passing of an optional "this" pointer.
- ClangUserExpression was extended to perform the
wrapping described above.
- ClangASTSource was changed to revert the changes
required by the hack.
- ClangExpressionParser, IRForTarget, and
ClangExpressionDeclMap were changed to handle
different manglings of ___clang_expr flexibly. This
meant no longer searching for a function called
___clang_expr, but rather looking for a function whose
name *contains* ___clang_expr.
- ClangExpressionParser and ClangExpressionDeclMap now
remember whether "this" is required, and know how to
look it up as necessary.
A few inheritance bugs remain, and I'm trying to resolve
these. But it is now possible to use "this" as well as
refer implicitly to member variables, when in the proper
context.
llvm-svn: 114384
2010-09-21 08:44:12 +08:00
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
|
Cleaned up the ABI::PrepareTrivialCall() function to take three argument
pointers:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t *arg1_ptr,
lldb::addr_t *arg2_ptr,
lldb::addr_t *arg3_ptr) const = 0;
Prior to this it was:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t arg,
lldb::addr_t *this_arg,
lldb::addr_t *cmd_arg) const = 0;
This was because the function that called this slowly added more features to
be able to call a C++ member function that might have a "this" pointer, and
then later added "self + cmd" support for objective C. Cleaning this code up
and the code that calls it makes it easier to implement the functions for
new targets.
The MacOSX_arm::PrepareTrivialCall() is now filled in and ready for testing.
llvm-svn: 131221
2011-05-12 10:14:56 +08:00
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
// First, align the SP
|
Cleaned up the ABI::PrepareTrivialCall() function to take three argument
pointers:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t *arg1_ptr,
lldb::addr_t *arg2_ptr,
lldb::addr_t *arg3_ptr) const = 0;
Prior to this it was:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t arg,
lldb::addr_t *this_arg,
lldb::addr_t *cmd_arg) const = 0;
This was because the function that called this slowly added more features to
be able to call a C++ member function that might have a "this" pointer, and
then later added "self + cmd" support for objective C. Cleaning this code up
and the code that calls it makes it easier to implement the functions for
new targets.
The MacOSX_arm::PrepareTrivialCall() is now filled in and ready for testing.
llvm-svn: 131221
2011-05-12 10:14:56 +08:00
|
|
|
|
2010-10-08 09:58:41 +08:00
|
|
|
if (log)
|
|
|
|
log->Printf("16-byte aligning SP: 0x%llx to 0x%llx", (uint64_t)sp, (uint64_t)(sp & ~0xfull));
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
sp &= ~(0xfull); // 16-byte alignment
|
|
|
|
|
Cleaned up the ABI::PrepareTrivialCall() function to take three argument
pointers:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t *arg1_ptr,
lldb::addr_t *arg2_ptr,
lldb::addr_t *arg3_ptr) const = 0;
Prior to this it was:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t arg,
lldb::addr_t *this_arg,
lldb::addr_t *cmd_arg) const = 0;
This was because the function that called this slowly added more features to
be able to call a C++ member function that might have a "this" pointer, and
then later added "self + cmd" support for objective C. Cleaning this code up
and the code that calls it makes it easier to implement the functions for
new targets.
The MacOSX_arm::PrepareTrivialCall() is now filled in and ready for testing.
llvm-svn: 131221
2011-05-12 10:14:56 +08:00
|
|
|
// The return address is pushed onto the stack (yes after the alignment...)
|
2010-06-09 00:52:24 +08:00
|
|
|
sp -= 8;
|
Cleaned up the ABI::PrepareTrivialCall() function to take three argument
pointers:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t *arg1_ptr,
lldb::addr_t *arg2_ptr,
lldb::addr_t *arg3_ptr) const = 0;
Prior to this it was:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t arg,
lldb::addr_t *this_arg,
lldb::addr_t *cmd_arg) const = 0;
This was because the function that called this slowly added more features to
be able to call a C++ member function that might have a "this" pointer, and
then later added "self + cmd" support for objective C. Cleaning this code up
and the code that calls it makes it easier to implement the functions for
new targets.
The MacOSX_arm::PrepareTrivialCall() is now filled in and ready for testing.
llvm-svn: 131221
2011-05-12 10:14:56 +08:00
|
|
|
|
2011-05-15 09:25:55 +08:00
|
|
|
RegisterValue reg_value;
|
Cleaned up the ABI::PrepareTrivialCall() function to take three argument
pointers:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t *arg1_ptr,
lldb::addr_t *arg2_ptr,
lldb::addr_t *arg3_ptr) const = 0;
Prior to this it was:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t arg,
lldb::addr_t *this_arg,
lldb::addr_t *cmd_arg) const = 0;
This was because the function that called this slowly added more features to
be able to call a C++ member function that might have a "this" pointer, and
then later added "self + cmd" support for objective C. Cleaning this code up
and the code that calls it makes it easier to implement the functions for
new targets.
The MacOSX_arm::PrepareTrivialCall() is now filled in and ready for testing.
llvm-svn: 131221
2011-05-12 10:14:56 +08:00
|
|
|
reg_value.SetUInt64 (return_addr);
|
|
|
|
|
2010-10-08 09:58:41 +08:00
|
|
|
if (log)
|
Cleaned up the ABI::PrepareTrivialCall() function to take three argument
pointers:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t *arg1_ptr,
lldb::addr_t *arg2_ptr,
lldb::addr_t *arg3_ptr) const = 0;
Prior to this it was:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t arg,
lldb::addr_t *this_arg,
lldb::addr_t *cmd_arg) const = 0;
This was because the function that called this slowly added more features to
be able to call a C++ member function that might have a "this" pointer, and
then later added "self + cmd" support for objective C. Cleaning this code up
and the code that calls it makes it easier to implement the functions for
new targets.
The MacOSX_arm::PrepareTrivialCall() is now filled in and ready for testing.
llvm-svn: 131221
2011-05-12 10:14:56 +08:00
|
|
|
log->Printf("Pushing the return address onto the stack: new SP 0x%llx, return address 0x%llx", (uint64_t)sp, (uint64_t)return_addr);
|
|
|
|
|
|
|
|
const RegisterInfo *pc_reg_info = reg_ctx->GetRegisterInfoByName("rip");
|
|
|
|
Error error (reg_ctx->WriteRegisterValueToMemory(pc_reg_info, sp, pc_reg_info->byte_size, reg_value));
|
|
|
|
if (error.Fail())
|
2010-06-09 00:52:24 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
// %rsp is set to the actual stack value.
|
|
|
|
|
2010-10-08 09:58:41 +08:00
|
|
|
if (log)
|
|
|
|
log->Printf("Writing SP (0x%llx) down", (uint64_t)sp);
|
|
|
|
|
2011-05-15 09:25:55 +08:00
|
|
|
if (!reg_ctx->WriteRegisterFromUnsigned (reg_ctx->GetRegisterInfoByName("rsp"), sp))
|
2010-06-09 00:52:24 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
// %rip is set to the address of the called function.
|
2010-10-08 09:58:41 +08:00
|
|
|
|
|
|
|
if (log)
|
Cleaned up the ABI::PrepareTrivialCall() function to take three argument
pointers:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t *arg1_ptr,
lldb::addr_t *arg2_ptr,
lldb::addr_t *arg3_ptr) const = 0;
Prior to this it was:
virtual bool
PrepareTrivialCall (Thread &thread,
lldb::addr_t sp,
lldb::addr_t functionAddress,
lldb::addr_t returnAddress,
lldb::addr_t arg,
lldb::addr_t *this_arg,
lldb::addr_t *cmd_arg) const = 0;
This was because the function that called this slowly added more features to
be able to call a C++ member function that might have a "this" pointer, and
then later added "self + cmd" support for objective C. Cleaning this code up
and the code that calls it makes it easier to implement the functions for
new targets.
The MacOSX_arm::PrepareTrivialCall() is now filled in and ready for testing.
llvm-svn: 131221
2011-05-12 10:14:56 +08:00
|
|
|
log->Printf("Writing new IP (0x%llx) down", (uint64_t)func_addr);
|
|
|
|
|
2011-05-15 09:25:55 +08:00
|
|
|
if (!reg_ctx->WriteRegisterFromUnsigned (pc_reg_info, func_addr))
|
2010-06-09 00:52:24 +08:00
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
static bool ReadIntegerArgument(Scalar &scalar,
|
|
|
|
unsigned int bit_width,
|
|
|
|
bool is_signed,
|
|
|
|
Thread &thread,
|
|
|
|
uint32_t *argument_register_ids,
|
|
|
|
unsigned int ¤t_argument_register,
|
|
|
|
addr_t ¤t_stack_argument)
|
|
|
|
{
|
|
|
|
if (bit_width > 64)
|
|
|
|
return false; // Scalar can't hold large integer arguments
|
|
|
|
|
|
|
|
if (current_argument_register < 6)
|
|
|
|
{
|
Added new lldb_private::Process memory read/write functions to stop a bunch
of duplicated code from appearing all over LLDB:
lldb::addr_t
Process::ReadPointerFromMemory (lldb::addr_t vm_addr, Error &error);
bool
Process::WritePointerToMemory (lldb::addr_t vm_addr, lldb::addr_t ptr_value, Error &error);
size_t
Process::ReadScalarIntegerFromMemory (lldb::addr_t addr, uint32_t byte_size, bool is_signed, Scalar &scalar, Error &error);
size_t
Process::WriteScalarToMemory (lldb::addr_t vm_addr, const Scalar &scalar, uint32_t size, Error &error);
in lldb_private::Process the following functions were renamed:
From:
uint64_t
Process::ReadUnsignedInteger (lldb::addr_t load_addr,
size_t byte_size,
Error &error);
To:
uint64_t
Process::ReadUnsignedIntegerFromMemory (lldb::addr_t load_addr,
size_t byte_size,
uint64_t fail_value,
Error &error);
Cleaned up a lot of code that was manually doing what the above functions do
to use the functions listed above.
Added the ability to get a scalar value as a buffer that can be written down
to a process (byte swapping the Scalar value if needed):
uint32_t
Scalar::GetAsMemoryData (void *dst,
uint32_t dst_len,
lldb::ByteOrder dst_byte_order,
Error &error) const;
The "dst_len" can be smaller that the size of the scalar and the least
significant bytes will be written. "dst_len" can also be larger and the
most significant bytes will be padded with zeroes.
Centralized the code that adds or removes address bits for callable and opcode
addresses into lldb_private::Target:
lldb::addr_t
Target::GetCallableLoadAddress (lldb::addr_t load_addr, AddressClass addr_class) const;
lldb::addr_t
Target::GetOpcodeLoadAddress (lldb::addr_t load_addr, AddressClass addr_class) const;
All necessary lldb_private::Address functions now use the target versions so
changes should only need to happen in one place if anything needs updating.
Fixed up a lot of places that were calling :
addr_t
Address::GetLoadAddress(Target*);
to call the Address::GetCallableLoadAddress() or Address::GetOpcodeLoadAddress()
as needed. There were many places in the breakpoint code where things could
go wrong for ARM if these weren't used.
llvm-svn: 131878
2011-05-23 06:46:53 +08:00
|
|
|
scalar = thread.GetRegisterContext()->ReadRegisterAsUnsigned(argument_register_ids[current_argument_register], 0);
|
2010-06-09 00:52:24 +08:00
|
|
|
current_argument_register++;
|
Added new lldb_private::Process memory read/write functions to stop a bunch
of duplicated code from appearing all over LLDB:
lldb::addr_t
Process::ReadPointerFromMemory (lldb::addr_t vm_addr, Error &error);
bool
Process::WritePointerToMemory (lldb::addr_t vm_addr, lldb::addr_t ptr_value, Error &error);
size_t
Process::ReadScalarIntegerFromMemory (lldb::addr_t addr, uint32_t byte_size, bool is_signed, Scalar &scalar, Error &error);
size_t
Process::WriteScalarToMemory (lldb::addr_t vm_addr, const Scalar &scalar, uint32_t size, Error &error);
in lldb_private::Process the following functions were renamed:
From:
uint64_t
Process::ReadUnsignedInteger (lldb::addr_t load_addr,
size_t byte_size,
Error &error);
To:
uint64_t
Process::ReadUnsignedIntegerFromMemory (lldb::addr_t load_addr,
size_t byte_size,
uint64_t fail_value,
Error &error);
Cleaned up a lot of code that was manually doing what the above functions do
to use the functions listed above.
Added the ability to get a scalar value as a buffer that can be written down
to a process (byte swapping the Scalar value if needed):
uint32_t
Scalar::GetAsMemoryData (void *dst,
uint32_t dst_len,
lldb::ByteOrder dst_byte_order,
Error &error) const;
The "dst_len" can be smaller that the size of the scalar and the least
significant bytes will be written. "dst_len" can also be larger and the
most significant bytes will be padded with zeroes.
Centralized the code that adds or removes address bits for callable and opcode
addresses into lldb_private::Target:
lldb::addr_t
Target::GetCallableLoadAddress (lldb::addr_t load_addr, AddressClass addr_class) const;
lldb::addr_t
Target::GetOpcodeLoadAddress (lldb::addr_t load_addr, AddressClass addr_class) const;
All necessary lldb_private::Address functions now use the target versions so
changes should only need to happen in one place if anything needs updating.
Fixed up a lot of places that were calling :
addr_t
Address::GetLoadAddress(Target*);
to call the Address::GetCallableLoadAddress() or Address::GetOpcodeLoadAddress()
as needed. There were many places in the breakpoint code where things could
go wrong for ARM if these weren't used.
llvm-svn: 131878
2011-05-23 06:46:53 +08:00
|
|
|
if (is_signed)
|
|
|
|
scalar.SignExtend (bit_width);
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
Added new lldb_private::Process memory read/write functions to stop a bunch
of duplicated code from appearing all over LLDB:
lldb::addr_t
Process::ReadPointerFromMemory (lldb::addr_t vm_addr, Error &error);
bool
Process::WritePointerToMemory (lldb::addr_t vm_addr, lldb::addr_t ptr_value, Error &error);
size_t
Process::ReadScalarIntegerFromMemory (lldb::addr_t addr, uint32_t byte_size, bool is_signed, Scalar &scalar, Error &error);
size_t
Process::WriteScalarToMemory (lldb::addr_t vm_addr, const Scalar &scalar, uint32_t size, Error &error);
in lldb_private::Process the following functions were renamed:
From:
uint64_t
Process::ReadUnsignedInteger (lldb::addr_t load_addr,
size_t byte_size,
Error &error);
To:
uint64_t
Process::ReadUnsignedIntegerFromMemory (lldb::addr_t load_addr,
size_t byte_size,
uint64_t fail_value,
Error &error);
Cleaned up a lot of code that was manually doing what the above functions do
to use the functions listed above.
Added the ability to get a scalar value as a buffer that can be written down
to a process (byte swapping the Scalar value if needed):
uint32_t
Scalar::GetAsMemoryData (void *dst,
uint32_t dst_len,
lldb::ByteOrder dst_byte_order,
Error &error) const;
The "dst_len" can be smaller that the size of the scalar and the least
significant bytes will be written. "dst_len" can also be larger and the
most significant bytes will be padded with zeroes.
Centralized the code that adds or removes address bits for callable and opcode
addresses into lldb_private::Target:
lldb::addr_t
Target::GetCallableLoadAddress (lldb::addr_t load_addr, AddressClass addr_class) const;
lldb::addr_t
Target::GetOpcodeLoadAddress (lldb::addr_t load_addr, AddressClass addr_class) const;
All necessary lldb_private::Address functions now use the target versions so
changes should only need to happen in one place if anything needs updating.
Fixed up a lot of places that were calling :
addr_t
Address::GetLoadAddress(Target*);
to call the Address::GetCallableLoadAddress() or Address::GetOpcodeLoadAddress()
as needed. There were many places in the breakpoint code where things could
go wrong for ARM if these weren't used.
llvm-svn: 131878
2011-05-23 06:46:53 +08:00
|
|
|
uint32_t byte_size = (bit_width + (8-1))/8;
|
2010-06-09 00:52:24 +08:00
|
|
|
Error error;
|
2012-02-21 08:09:25 +08:00
|
|
|
if (thread.GetProcess()->ReadScalarIntegerFromMemory(current_stack_argument, byte_size, is_signed, scalar, error))
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
Added new lldb_private::Process memory read/write functions to stop a bunch
of duplicated code from appearing all over LLDB:
lldb::addr_t
Process::ReadPointerFromMemory (lldb::addr_t vm_addr, Error &error);
bool
Process::WritePointerToMemory (lldb::addr_t vm_addr, lldb::addr_t ptr_value, Error &error);
size_t
Process::ReadScalarIntegerFromMemory (lldb::addr_t addr, uint32_t byte_size, bool is_signed, Scalar &scalar, Error &error);
size_t
Process::WriteScalarToMemory (lldb::addr_t vm_addr, const Scalar &scalar, uint32_t size, Error &error);
in lldb_private::Process the following functions were renamed:
From:
uint64_t
Process::ReadUnsignedInteger (lldb::addr_t load_addr,
size_t byte_size,
Error &error);
To:
uint64_t
Process::ReadUnsignedIntegerFromMemory (lldb::addr_t load_addr,
size_t byte_size,
uint64_t fail_value,
Error &error);
Cleaned up a lot of code that was manually doing what the above functions do
to use the functions listed above.
Added the ability to get a scalar value as a buffer that can be written down
to a process (byte swapping the Scalar value if needed):
uint32_t
Scalar::GetAsMemoryData (void *dst,
uint32_t dst_len,
lldb::ByteOrder dst_byte_order,
Error &error) const;
The "dst_len" can be smaller that the size of the scalar and the least
significant bytes will be written. "dst_len" can also be larger and the
most significant bytes will be padded with zeroes.
Centralized the code that adds or removes address bits for callable and opcode
addresses into lldb_private::Target:
lldb::addr_t
Target::GetCallableLoadAddress (lldb::addr_t load_addr, AddressClass addr_class) const;
lldb::addr_t
Target::GetOpcodeLoadAddress (lldb::addr_t load_addr, AddressClass addr_class) const;
All necessary lldb_private::Address functions now use the target versions so
changes should only need to happen in one place if anything needs updating.
Fixed up a lot of places that were calling :
addr_t
Address::GetLoadAddress(Target*);
to call the Address::GetCallableLoadAddress() or Address::GetOpcodeLoadAddress()
as needed. There were many places in the breakpoint code where things could
go wrong for ARM if these weren't used.
llvm-svn: 131878
2011-05-23 06:46:53 +08:00
|
|
|
current_stack_argument += byte_size;
|
|
|
|
return true;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
Added new lldb_private::Process memory read/write functions to stop a bunch
of duplicated code from appearing all over LLDB:
lldb::addr_t
Process::ReadPointerFromMemory (lldb::addr_t vm_addr, Error &error);
bool
Process::WritePointerToMemory (lldb::addr_t vm_addr, lldb::addr_t ptr_value, Error &error);
size_t
Process::ReadScalarIntegerFromMemory (lldb::addr_t addr, uint32_t byte_size, bool is_signed, Scalar &scalar, Error &error);
size_t
Process::WriteScalarToMemory (lldb::addr_t vm_addr, const Scalar &scalar, uint32_t size, Error &error);
in lldb_private::Process the following functions were renamed:
From:
uint64_t
Process::ReadUnsignedInteger (lldb::addr_t load_addr,
size_t byte_size,
Error &error);
To:
uint64_t
Process::ReadUnsignedIntegerFromMemory (lldb::addr_t load_addr,
size_t byte_size,
uint64_t fail_value,
Error &error);
Cleaned up a lot of code that was manually doing what the above functions do
to use the functions listed above.
Added the ability to get a scalar value as a buffer that can be written down
to a process (byte swapping the Scalar value if needed):
uint32_t
Scalar::GetAsMemoryData (void *dst,
uint32_t dst_len,
lldb::ByteOrder dst_byte_order,
Error &error) const;
The "dst_len" can be smaller that the size of the scalar and the least
significant bytes will be written. "dst_len" can also be larger and the
most significant bytes will be padded with zeroes.
Centralized the code that adds or removes address bits for callable and opcode
addresses into lldb_private::Target:
lldb::addr_t
Target::GetCallableLoadAddress (lldb::addr_t load_addr, AddressClass addr_class) const;
lldb::addr_t
Target::GetOpcodeLoadAddress (lldb::addr_t load_addr, AddressClass addr_class) const;
All necessary lldb_private::Address functions now use the target versions so
changes should only need to happen in one place if anything needs updating.
Fixed up a lot of places that were calling :
addr_t
Address::GetLoadAddress(Target*);
to call the Address::GetCallableLoadAddress() or Address::GetOpcodeLoadAddress()
as needed. There were many places in the breakpoint code where things could
go wrong for ARM if these weren't used.
llvm-svn: 131878
2011-05-23 06:46:53 +08:00
|
|
|
return false;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
ABISysV_x86_64::GetArgumentValues (Thread &thread,
|
|
|
|
ValueList &values) const
|
|
|
|
{
|
|
|
|
unsigned int num_values = values.GetSize();
|
|
|
|
unsigned int value_index;
|
|
|
|
|
|
|
|
// For now, assume that the types in the AST values come from the Target's
|
|
|
|
// scratch AST.
|
|
|
|
|
|
|
|
clang::ASTContext *ast_context = thread.CalculateTarget()->GetScratchClangASTContext()->getASTContext();
|
|
|
|
|
|
|
|
// Extract the register context so we can read arguments from registers
|
|
|
|
|
2011-01-07 06:15:06 +08:00
|
|
|
RegisterContext *reg_ctx = thread.GetRegisterContext().get();
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
if (!reg_ctx)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// Get the pointer to the first stack argument so we have a place to start
|
|
|
|
// when reading data
|
|
|
|
|
|
|
|
addr_t sp = reg_ctx->GetSP(0);
|
|
|
|
|
|
|
|
if (!sp)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
addr_t current_stack_argument = sp + 8; // jump over return address
|
|
|
|
|
|
|
|
uint32_t argument_register_ids[6];
|
|
|
|
|
2010-09-10 15:49:16 +08:00
|
|
|
argument_register_ids[0] = reg_ctx->GetRegisterInfoByName("rdi", 0)->kinds[eRegisterKindLLDB];
|
|
|
|
argument_register_ids[1] = reg_ctx->GetRegisterInfoByName("rsi", 0)->kinds[eRegisterKindLLDB];
|
|
|
|
argument_register_ids[2] = reg_ctx->GetRegisterInfoByName("rdx", 0)->kinds[eRegisterKindLLDB];
|
|
|
|
argument_register_ids[3] = reg_ctx->GetRegisterInfoByName("rcx", 0)->kinds[eRegisterKindLLDB];
|
|
|
|
argument_register_ids[4] = reg_ctx->GetRegisterInfoByName("r8", 0)->kinds[eRegisterKindLLDB];
|
|
|
|
argument_register_ids[5] = reg_ctx->GetRegisterInfoByName("r9", 0)->kinds[eRegisterKindLLDB];
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
unsigned int current_argument_register = 0;
|
|
|
|
|
|
|
|
for (value_index = 0;
|
|
|
|
value_index < num_values;
|
|
|
|
++value_index)
|
|
|
|
{
|
|
|
|
Value *value = values.GetValueAtIndex(value_index);
|
|
|
|
|
|
|
|
if (!value)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
// We currently only support extracting values with Clang QualTypes.
|
|
|
|
// Do we care about others?
|
|
|
|
switch (value->GetContextType())
|
|
|
|
{
|
|
|
|
default:
|
|
|
|
return false;
|
2010-11-13 11:52:47 +08:00
|
|
|
case Value::eContextTypeClangType:
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2010-09-29 09:12:09 +08:00
|
|
|
void *value_type = value->GetClangType();
|
2010-06-09 00:52:24 +08:00
|
|
|
bool is_signed;
|
|
|
|
|
|
|
|
if (ClangASTContext::IsIntegerType (value_type, is_signed))
|
|
|
|
{
|
2010-08-03 08:35:52 +08:00
|
|
|
size_t bit_width = ClangASTType::GetClangTypeBitWidth(ast_context, value_type);
|
2010-06-09 00:52:24 +08:00
|
|
|
|
|
|
|
ReadIntegerArgument(value->GetScalar(),
|
|
|
|
bit_width,
|
|
|
|
is_signed,
|
|
|
|
thread,
|
|
|
|
argument_register_ids,
|
|
|
|
current_argument_register,
|
|
|
|
current_stack_argument);
|
|
|
|
}
|
|
|
|
else if (ClangASTContext::IsPointerType (value_type))
|
|
|
|
{
|
|
|
|
ReadIntegerArgument(value->GetScalar(),
|
|
|
|
64,
|
|
|
|
false,
|
|
|
|
thread,
|
|
|
|
argument_register_ids,
|
|
|
|
current_argument_register,
|
|
|
|
current_stack_argument);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2012-09-27 09:15:29 +08:00
|
|
|
Error
|
|
|
|
ABISysV_x86_64::SetReturnValueObject(lldb::StackFrameSP &frame_sp, lldb::ValueObjectSP &new_value_sp)
|
|
|
|
{
|
|
|
|
Error error;
|
|
|
|
if (!new_value_sp)
|
|
|
|
{
|
|
|
|
error.SetErrorString("Empty value object for return value.");
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
clang_type_t value_type = new_value_sp->GetClangType();
|
|
|
|
if (!value_type)
|
|
|
|
{
|
|
|
|
error.SetErrorString ("Null clang type for return value.");
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
clang::ASTContext *ast_context = new_value_sp->GetClangAST();
|
|
|
|
if (!ast_context)
|
|
|
|
{
|
|
|
|
error.SetErrorString ("Null clang AST for return value.");
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
Thread *thread = frame_sp->GetThread().get();
|
|
|
|
|
|
|
|
bool is_signed;
|
|
|
|
uint32_t count;
|
|
|
|
bool is_complex;
|
|
|
|
|
|
|
|
RegisterContext *reg_ctx = thread->GetRegisterContext().get();
|
|
|
|
|
|
|
|
bool set_it_simple = false;
|
|
|
|
if (ClangASTContext::IsIntegerType (value_type, is_signed) || ClangASTContext::IsPointerType(value_type))
|
|
|
|
{
|
|
|
|
const RegisterInfo *reg_info = reg_ctx->GetRegisterInfoByName("rax", 0);
|
|
|
|
|
|
|
|
DataExtractor data;
|
|
|
|
size_t num_bytes = new_value_sp->GetData(data);
|
|
|
|
uint32_t offset = 0;
|
|
|
|
if (num_bytes <= 64)
|
|
|
|
{
|
|
|
|
uint64_t raw_value = data.GetMaxU64(&offset, num_bytes);
|
|
|
|
|
|
|
|
if (reg_ctx->WriteRegisterFromUnsigned (reg_info, raw_value))
|
|
|
|
set_it_simple = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
error.SetErrorString("We don't support returning longer than 64 bit integer values at present.");
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
else if (ClangASTContext::IsFloatingPointType (value_type, count, is_complex))
|
|
|
|
{
|
|
|
|
if (is_complex)
|
|
|
|
error.SetErrorString ("We don't support returning complex values at present");
|
|
|
|
else
|
|
|
|
{
|
|
|
|
size_t bit_width = ClangASTType::GetClangTypeBitWidth(ast_context, value_type);
|
|
|
|
if (bit_width <= 64)
|
|
|
|
{
|
|
|
|
const RegisterInfo *xmm0_info = reg_ctx->GetRegisterInfoByName("xmm0", 0);
|
|
|
|
RegisterValue xmm0_value;
|
|
|
|
DataExtractor data;
|
|
|
|
size_t num_bytes = new_value_sp->GetData(data);
|
|
|
|
|
|
|
|
unsigned char buffer[16];
|
|
|
|
ByteOrder byte_order = data.GetByteOrder();
|
|
|
|
uint32_t return_bytes;
|
|
|
|
|
|
|
|
return_bytes = data.CopyByteOrderedData (0, num_bytes, buffer, 16, byte_order);
|
|
|
|
xmm0_value.SetBytes(buffer, 16, byte_order);
|
|
|
|
reg_ctx->WriteRegister(xmm0_info, xmm0_value);
|
|
|
|
set_it_simple = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// FIXME - don't know how to do 80 bit long doubles yet.
|
|
|
|
error.SetErrorString ("We don't support returning float values > 64 bits at present");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!set_it_simple)
|
|
|
|
{
|
|
|
|
// Okay we've got a structure or something that doesn't fit in a simple register.
|
|
|
|
// We should figure out where it really goes, but we don't support this yet.
|
|
|
|
error.SetErrorString ("We only support setting simple integer and float return types at present.");
|
|
|
|
}
|
|
|
|
|
|
|
|
return error;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-12-23 03:12:40 +08:00
|
|
|
ValueObjectSP
|
|
|
|
ABISysV_x86_64::GetReturnValueObjectSimple (Thread &thread,
|
|
|
|
ClangASTType &ast_type) const
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2011-12-23 03:12:40 +08:00
|
|
|
ValueObjectSP return_valobj_sp;
|
|
|
|
Value value;
|
|
|
|
|
|
|
|
clang_type_t value_type = ast_type.GetOpaqueQualType();
|
|
|
|
if (!value_type)
|
|
|
|
return return_valobj_sp;
|
|
|
|
|
|
|
|
clang::ASTContext *ast_context = ast_type.GetASTContext();
|
|
|
|
if (!ast_context)
|
|
|
|
return return_valobj_sp;
|
|
|
|
|
|
|
|
value.SetContext (Value::eContextTypeClangType, value_type);
|
|
|
|
|
|
|
|
RegisterContext *reg_ctx = thread.GetRegisterContext().get();
|
|
|
|
if (!reg_ctx)
|
|
|
|
return return_valobj_sp;
|
|
|
|
|
|
|
|
bool is_signed;
|
|
|
|
bool is_complex;
|
|
|
|
uint32_t count;
|
|
|
|
|
|
|
|
if (ClangASTContext::IsIntegerType (value_type, is_signed))
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2011-12-23 03:12:40 +08:00
|
|
|
// For now, assume that the types in the AST values come from the Target's
|
|
|
|
// scratch AST.
|
|
|
|
|
|
|
|
|
|
|
|
// Extract the register context so we can read arguments from registers
|
|
|
|
|
|
|
|
size_t bit_width = ClangASTType::GetClangTypeBitWidth(ast_context, value_type);
|
2012-09-27 09:15:29 +08:00
|
|
|
uint64_t raw_value = thread.GetRegisterContext()->ReadRegisterAsUnsigned(reg_ctx->GetRegisterInfoByName("rax", 0), 0);
|
2011-12-23 03:12:40 +08:00
|
|
|
|
|
|
|
switch (bit_width)
|
|
|
|
{
|
2010-06-09 00:52:24 +08:00
|
|
|
default:
|
2011-12-23 03:12:40 +08:00
|
|
|
case 128:
|
|
|
|
// Scalar can't hold 128-bit literals, so we don't handle this
|
|
|
|
return return_valobj_sp;
|
|
|
|
case 64:
|
|
|
|
if (is_signed)
|
2012-09-27 09:15:29 +08:00
|
|
|
value.GetScalar() = (int64_t)(raw_value);
|
2011-12-23 03:12:40 +08:00
|
|
|
else
|
2012-09-27 09:15:29 +08:00
|
|
|
value.GetScalar() = (uint64_t)(raw_value);
|
2011-12-23 03:12:40 +08:00
|
|
|
break;
|
|
|
|
case 32:
|
|
|
|
if (is_signed)
|
2012-09-27 09:15:29 +08:00
|
|
|
value.GetScalar() = (int32_t)(raw_value & 0xffffffff);
|
2011-12-23 03:12:40 +08:00
|
|
|
else
|
2012-09-27 09:15:29 +08:00
|
|
|
value.GetScalar() = (uint32_t)(raw_value & 0xffffffff);
|
2011-12-23 03:12:40 +08:00
|
|
|
break;
|
|
|
|
case 16:
|
|
|
|
if (is_signed)
|
2012-09-27 09:15:29 +08:00
|
|
|
value.GetScalar() = (int16_t)(raw_value & 0xffff);
|
2011-12-23 03:12:40 +08:00
|
|
|
else
|
2012-09-27 09:15:29 +08:00
|
|
|
value.GetScalar() = (uint16_t)(raw_value & 0xffff);
|
2011-12-23 03:12:40 +08:00
|
|
|
break;
|
|
|
|
case 8:
|
|
|
|
if (is_signed)
|
2012-09-27 09:15:29 +08:00
|
|
|
value.GetScalar() = (int8_t)(raw_value & 0xff);
|
2011-12-23 03:12:40 +08:00
|
|
|
else
|
2012-09-27 09:15:29 +08:00
|
|
|
value.GetScalar() = (uint8_t)(raw_value & 0xff);
|
2011-12-23 03:12:40 +08:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (ClangASTContext::IsFloatingPointType(value_type, count, is_complex))
|
|
|
|
{
|
|
|
|
// Don't handle complex yet.
|
|
|
|
if (is_complex)
|
|
|
|
return return_valobj_sp;
|
|
|
|
|
|
|
|
size_t bit_width = ClangASTType::GetClangTypeBitWidth(ast_context, value_type);
|
|
|
|
if (bit_width <= 64)
|
|
|
|
{
|
|
|
|
const RegisterInfo *xmm0_info = reg_ctx->GetRegisterInfoByName("xmm0", 0);
|
|
|
|
RegisterValue xmm0_value;
|
|
|
|
if (reg_ctx->ReadRegister (xmm0_info, xmm0_value))
|
|
|
|
{
|
|
|
|
DataExtractor data;
|
|
|
|
if (xmm0_value.GetData(data))
|
|
|
|
{
|
|
|
|
uint32_t offset = 0;
|
|
|
|
switch (bit_width)
|
|
|
|
{
|
|
|
|
default:
|
|
|
|
return return_valobj_sp;
|
|
|
|
case 32:
|
|
|
|
value.GetScalar() = (float) data.GetFloat(&offset);
|
|
|
|
break;
|
|
|
|
case 64:
|
|
|
|
value.GetScalar() = (double) data.GetDouble(&offset);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (bit_width == 128)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2011-12-23 03:12:40 +08:00
|
|
|
// FIXME: x86_64 returns long doubles in stmm0, which is in some 80 bit long double
|
|
|
|
// format, and so we'll have to write some code to convert that into 128 bit long doubles.
|
|
|
|
// const RegisterInfo *st0_info = reg_ctx->GetRegisterInfoByName("stmm0", 0);
|
|
|
|
// RegisterValue st0_value;
|
|
|
|
// if (reg_ctx->ReadRegister (st0_info, st0_value))
|
|
|
|
// {
|
|
|
|
// DataExtractor data;
|
|
|
|
// if (st0_value.GetData(data))
|
|
|
|
// {
|
|
|
|
// uint32_t offset = 0;
|
|
|
|
// value.GetScalar() = (long double) data.GetLongDouble (&offset);
|
|
|
|
// return true;
|
|
|
|
// }
|
|
|
|
// }
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2011-12-23 03:12:40 +08:00
|
|
|
return return_valobj_sp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (ClangASTContext::IsPointerType (value_type))
|
|
|
|
{
|
|
|
|
unsigned rax_id = reg_ctx->GetRegisterInfoByName("rax", 0)->kinds[eRegisterKindLLDB];
|
|
|
|
value.GetScalar() = (uint64_t)thread.GetRegisterContext()->ReadRegisterAsUnsigned(rax_id, 0);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
return return_valobj_sp;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we get here, we have a valid Value, so make our ValueObject out of it:
|
|
|
|
|
|
|
|
return_valobj_sp = ValueObjectConstResult::Create(
|
|
|
|
thread.GetStackFrameAtIndex(0).get(),
|
|
|
|
ast_type.GetASTContext(),
|
|
|
|
value,
|
|
|
|
ConstString(""));
|
|
|
|
return return_valobj_sp;
|
|
|
|
}
|
|
|
|
|
|
|
|
ValueObjectSP
|
2012-02-21 08:09:25 +08:00
|
|
|
ABISysV_x86_64::GetReturnValueObjectImpl (Thread &thread, ClangASTType &ast_type) const
|
2011-12-23 03:12:40 +08:00
|
|
|
{
|
|
|
|
ValueObjectSP return_valobj_sp;
|
|
|
|
|
2012-02-21 08:09:25 +08:00
|
|
|
ExecutionContext exe_ctx (thread.shared_from_this());
|
2011-12-23 03:12:40 +08:00
|
|
|
return_valobj_sp = GetReturnValueObjectSimple(thread, ast_type);
|
|
|
|
if (return_valobj_sp)
|
|
|
|
return return_valobj_sp;
|
|
|
|
|
|
|
|
clang_type_t ret_value_type = ast_type.GetOpaqueQualType();
|
|
|
|
if (!ret_value_type)
|
|
|
|
return return_valobj_sp;
|
|
|
|
|
|
|
|
clang::ASTContext *ast_context = ast_type.GetASTContext();
|
|
|
|
if (!ast_context)
|
|
|
|
return return_valobj_sp;
|
|
|
|
|
|
|
|
RegisterContextSP reg_ctx_sp = thread.GetRegisterContext();
|
|
|
|
if (!reg_ctx_sp)
|
|
|
|
return return_valobj_sp;
|
|
|
|
|
|
|
|
size_t bit_width = ClangASTType::GetClangTypeBitWidth(ast_context, ret_value_type);
|
|
|
|
if (ClangASTContext::IsAggregateType(ret_value_type))
|
|
|
|
{
|
2012-02-21 08:09:25 +08:00
|
|
|
Target *target = exe_ctx.GetTargetPtr();
|
2011-12-23 03:12:40 +08:00
|
|
|
bool is_memory = true;
|
|
|
|
if (bit_width <= 128)
|
|
|
|
{
|
2012-02-21 08:09:25 +08:00
|
|
|
ByteOrder target_byte_order = target->GetArchitecture().GetByteOrder();
|
2011-12-23 03:12:40 +08:00
|
|
|
DataBufferSP data_sp (new DataBufferHeap(16, 0));
|
|
|
|
DataExtractor return_ext (data_sp,
|
|
|
|
target_byte_order,
|
2012-02-21 08:09:25 +08:00
|
|
|
target->GetArchitecture().GetAddressByteSize());
|
2011-12-23 03:12:40 +08:00
|
|
|
|
|
|
|
const RegisterInfo *rax_info = reg_ctx_sp->GetRegisterInfoByName("rax", 0);
|
|
|
|
const RegisterInfo *rdx_info = reg_ctx_sp->GetRegisterInfoByName("rdx", 0);
|
|
|
|
const RegisterInfo *xmm0_info = reg_ctx_sp->GetRegisterInfoByName("xmm0", 0);
|
|
|
|
const RegisterInfo *xmm1_info = reg_ctx_sp->GetRegisterInfoByName("xmm1", 0);
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2011-12-23 03:12:40 +08:00
|
|
|
RegisterValue rax_value, rdx_value, xmm0_value, xmm1_value;
|
|
|
|
reg_ctx_sp->ReadRegister (rax_info, rax_value);
|
|
|
|
reg_ctx_sp->ReadRegister (rdx_info, rdx_value);
|
|
|
|
reg_ctx_sp->ReadRegister (xmm0_info, xmm0_value);
|
|
|
|
reg_ctx_sp->ReadRegister (xmm1_info, xmm1_value);
|
|
|
|
|
|
|
|
DataExtractor rax_data, rdx_data, xmm0_data, xmm1_data;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2011-12-23 03:12:40 +08:00
|
|
|
rax_value.GetData(rax_data);
|
|
|
|
rdx_value.GetData(rdx_data);
|
|
|
|
xmm0_value.GetData(xmm0_data);
|
|
|
|
xmm1_value.GetData(xmm1_data);
|
|
|
|
|
|
|
|
uint32_t fp_bytes = 0; // Tracks how much of the xmm registers we've consumed so far
|
|
|
|
uint32_t integer_bytes = 0; // Tracks how much of the rax/rds registers we've consumed so far
|
|
|
|
|
|
|
|
uint32_t num_children = ClangASTContext::GetNumFields (ast_context, ret_value_type);
|
|
|
|
|
|
|
|
// Since we are in the small struct regime, assume we are not in memory.
|
|
|
|
is_memory = false;
|
|
|
|
|
|
|
|
for (uint32_t idx = 0; idx < num_children; idx++)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2011-12-23 03:12:40 +08:00
|
|
|
std::string name;
|
2012-08-01 07:39:10 +08:00
|
|
|
uint64_t field_bit_offset = 0;
|
2011-12-23 03:12:40 +08:00
|
|
|
bool is_signed;
|
|
|
|
bool is_complex;
|
|
|
|
uint32_t count;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2012-08-01 07:39:10 +08:00
|
|
|
clang_type_t field_clang_type = ClangASTContext::GetFieldAtIndex (ast_context, ret_value_type, idx, name, &field_bit_offset, NULL, NULL);
|
2011-12-23 03:12:40 +08:00
|
|
|
size_t field_bit_width = ClangASTType::GetClangTypeBitWidth(ast_context, field_clang_type);
|
|
|
|
|
|
|
|
// If there are any unaligned fields, this is stored in memory.
|
|
|
|
if (field_bit_offset % field_bit_width != 0)
|
|
|
|
{
|
|
|
|
is_memory = true;
|
|
|
|
break;
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2011-12-23 03:12:40 +08:00
|
|
|
uint32_t field_byte_width = field_bit_width/8;
|
|
|
|
uint32_t field_byte_offset = field_bit_offset/8;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2011-12-23 03:12:40 +08:00
|
|
|
|
|
|
|
DataExtractor *copy_from_extractor = NULL;
|
|
|
|
uint32_t copy_from_offset = 0;
|
2010-06-09 00:52:24 +08:00
|
|
|
|
2011-12-23 03:12:40 +08:00
|
|
|
if (ClangASTContext::IsIntegerType (field_clang_type, is_signed) || ClangASTContext::IsPointerType (field_clang_type))
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2011-12-23 03:12:40 +08:00
|
|
|
if (integer_bytes < 8)
|
|
|
|
{
|
|
|
|
if (integer_bytes + field_byte_width <= 8)
|
|
|
|
{
|
|
|
|
// This is in RAX, copy from register to our result structure:
|
|
|
|
copy_from_extractor = &rax_data;
|
|
|
|
copy_from_offset = integer_bytes;
|
|
|
|
integer_bytes += field_byte_width;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// The next field wouldn't fit in the remaining space, so we pushed it to rdx.
|
|
|
|
copy_from_extractor = &rdx_data;
|
|
|
|
copy_from_offset = 0;
|
|
|
|
integer_bytes = 8 + field_byte_width;
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (integer_bytes + field_byte_width <= 16)
|
|
|
|
{
|
|
|
|
copy_from_extractor = &rdx_data;
|
|
|
|
copy_from_offset = integer_bytes - 8;
|
|
|
|
integer_bytes += field_byte_width;
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
else
|
2011-12-23 03:12:40 +08:00
|
|
|
{
|
|
|
|
// The last field didn't fit. I can't see how that would happen w/o the overall size being
|
|
|
|
// greater than 16 bytes. For now, return a NULL return value object.
|
|
|
|
return return_valobj_sp;
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
2011-12-23 03:12:40 +08:00
|
|
|
else if (ClangASTContext::IsFloatingPointType (field_clang_type, count, is_complex))
|
|
|
|
{
|
|
|
|
// Structs with long doubles are always passed in memory.
|
|
|
|
if (field_bit_width == 128)
|
|
|
|
{
|
|
|
|
is_memory = true;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
else if (field_bit_width == 64)
|
|
|
|
{
|
|
|
|
// These have to be in a single xmm register.
|
|
|
|
if (fp_bytes == 0)
|
|
|
|
copy_from_extractor = &xmm0_data;
|
|
|
|
else
|
|
|
|
copy_from_extractor = &xmm1_data;
|
|
|
|
|
|
|
|
copy_from_offset = 0;
|
|
|
|
fp_bytes += field_byte_width;
|
|
|
|
}
|
|
|
|
else if (field_bit_width == 32)
|
|
|
|
{
|
|
|
|
// This one is kind of complicated. If we are in an "eightbyte" with another float, we'll
|
|
|
|
// be stuffed into an xmm register with it. If we are in an "eightbyte" with one or more ints,
|
|
|
|
// then we will be stuffed into the appropriate GPR with them.
|
|
|
|
bool in_gpr;
|
|
|
|
if (field_byte_offset % 8 == 0)
|
|
|
|
{
|
|
|
|
// We are at the beginning of one of the eightbytes, so check the next element (if any)
|
|
|
|
if (idx == num_children - 1)
|
2011-12-23 08:30:10 +08:00
|
|
|
in_gpr = false;
|
2011-12-23 03:12:40 +08:00
|
|
|
else
|
|
|
|
{
|
2012-08-01 07:39:10 +08:00
|
|
|
uint64_t next_field_bit_offset = 0;
|
|
|
|
clang_type_t next_field_clang_type = ClangASTContext::GetFieldAtIndex (ast_context,
|
2011-12-23 03:12:40 +08:00
|
|
|
ret_value_type,
|
|
|
|
idx + 1,
|
|
|
|
name,
|
2012-08-01 07:39:10 +08:00
|
|
|
&next_field_bit_offset,
|
|
|
|
NULL,
|
|
|
|
NULL);
|
2011-12-23 03:12:40 +08:00
|
|
|
if (ClangASTContext::IsIntegerType (next_field_clang_type, is_signed))
|
|
|
|
in_gpr = true;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
copy_from_offset = 0;
|
|
|
|
in_gpr = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
else if (field_byte_offset % 4 == 0)
|
|
|
|
{
|
|
|
|
// We are inside of an eightbyte, so see if the field before us is floating point:
|
|
|
|
// This could happen if somebody put padding in the structure.
|
|
|
|
if (idx == 0)
|
|
|
|
in_gpr = false;
|
|
|
|
else
|
|
|
|
{
|
2012-08-01 07:39:10 +08:00
|
|
|
uint64_t prev_field_bit_offset = 0;
|
2011-12-23 03:12:40 +08:00
|
|
|
clang_type_t prev_field_clang_type = ClangASTContext::GetFieldAtIndex (ast_context,
|
|
|
|
ret_value_type,
|
|
|
|
idx - 1,
|
|
|
|
name,
|
2012-08-01 07:39:10 +08:00
|
|
|
&prev_field_bit_offset,
|
|
|
|
NULL,
|
|
|
|
NULL);
|
2011-12-23 03:12:40 +08:00
|
|
|
if (ClangASTContext::IsIntegerType (prev_field_clang_type, is_signed))
|
|
|
|
in_gpr = true;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
copy_from_offset = 4;
|
|
|
|
in_gpr = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
is_memory = true;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Okay, we've figured out whether we are in GPR or XMM, now figure out which one.
|
|
|
|
if (in_gpr)
|
|
|
|
{
|
|
|
|
if (integer_bytes < 8)
|
|
|
|
{
|
|
|
|
// This is in RAX, copy from register to our result structure:
|
|
|
|
copy_from_extractor = &rax_data;
|
|
|
|
copy_from_offset = integer_bytes;
|
|
|
|
integer_bytes += field_byte_width;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
copy_from_extractor = &rdx_data;
|
|
|
|
copy_from_offset = integer_bytes - 8;
|
|
|
|
integer_bytes += field_byte_width;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (fp_bytes < 8)
|
|
|
|
copy_from_extractor = &xmm0_data;
|
|
|
|
else
|
|
|
|
copy_from_extractor = &xmm1_data;
|
|
|
|
|
|
|
|
fp_bytes += field_byte_width;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2011-12-23 08:57:42 +08:00
|
|
|
|
|
|
|
// These two tests are just sanity checks. If I somehow get the
|
|
|
|
// type calculation wrong above it is better to just return nothing
|
2012-09-18 08:00:25 +08:00
|
|
|
// than to assert or crash.
|
2011-12-23 03:12:40 +08:00
|
|
|
if (!copy_from_extractor)
|
|
|
|
return return_valobj_sp;
|
2011-12-23 08:57:42 +08:00
|
|
|
if (copy_from_offset + field_byte_width > copy_from_extractor->GetByteSize())
|
|
|
|
return return_valobj_sp;
|
2011-12-23 03:12:40 +08:00
|
|
|
|
|
|
|
copy_from_extractor->CopyByteOrderedData (copy_from_offset,
|
|
|
|
field_byte_width,
|
|
|
|
data_sp->GetBytes() + field_byte_offset,
|
|
|
|
field_byte_width,
|
|
|
|
target_byte_order);
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
2011-12-23 03:12:40 +08:00
|
|
|
|
|
|
|
if (!is_memory)
|
2010-06-09 00:52:24 +08:00
|
|
|
{
|
2011-12-23 03:12:40 +08:00
|
|
|
// The result is in our data buffer. Let's make a variable object out of it:
|
|
|
|
return_valobj_sp = ValueObjectConstResult::Create (&thread,
|
|
|
|
ast_context,
|
|
|
|
ret_value_type,
|
|
|
|
ConstString(""),
|
|
|
|
return_ext);
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
}
|
2011-12-23 03:12:40 +08:00
|
|
|
|
|
|
|
if (is_memory)
|
|
|
|
{
|
|
|
|
unsigned rax_id = reg_ctx_sp->GetRegisterInfoByName("rax", 0)->kinds[eRegisterKindLLDB];
|
|
|
|
lldb::addr_t storage_addr = (uint64_t)thread.GetRegisterContext()->ReadRegisterAsUnsigned(rax_id, 0);
|
|
|
|
return_valobj_sp = ValueObjectMemory::Create (&thread,
|
|
|
|
"",
|
|
|
|
Address (storage_addr, NULL),
|
|
|
|
ast_type);
|
|
|
|
}
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
2011-12-23 03:12:40 +08:00
|
|
|
|
|
|
|
return return_valobj_sp;
|
2010-06-09 00:52:24 +08:00
|
|
|
}
|
|
|
|
|
2011-05-12 02:39:18 +08:00
|
|
|
bool
|
|
|
|
ABISysV_x86_64::CreateFunctionEntryUnwindPlan (UnwindPlan &unwind_plan)
|
|
|
|
{
|
|
|
|
uint32_t reg_kind = unwind_plan.GetRegisterKind();
|
|
|
|
uint32_t sp_reg_num = LLDB_INVALID_REGNUM;
|
|
|
|
uint32_t pc_reg_num = LLDB_INVALID_REGNUM;
|
|
|
|
|
|
|
|
switch (reg_kind)
|
|
|
|
{
|
|
|
|
case eRegisterKindDWARF:
|
|
|
|
case eRegisterKindGCC:
|
|
|
|
sp_reg_num = gcc_dwarf_rsp;
|
|
|
|
pc_reg_num = gcc_dwarf_rip;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eRegisterKindGDB:
|
|
|
|
sp_reg_num = gdb_rsp;
|
|
|
|
pc_reg_num = gdb_rip;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eRegisterKindGeneric:
|
|
|
|
sp_reg_num = LLDB_REGNUM_GENERIC_SP;
|
|
|
|
pc_reg_num = LLDB_REGNUM_GENERIC_PC;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (sp_reg_num == LLDB_INVALID_REGNUM ||
|
|
|
|
pc_reg_num == LLDB_INVALID_REGNUM)
|
|
|
|
return false;
|
|
|
|
|
2012-07-14 12:52:53 +08:00
|
|
|
UnwindPlan::RowSP row(new UnwindPlan::Row);
|
|
|
|
row->SetCFARegister (sp_reg_num);
|
|
|
|
row->SetCFAOffset (8);
|
|
|
|
row->SetRegisterLocationToAtCFAPlusOffset(pc_reg_num, -8, false);
|
2011-05-12 02:39:18 +08:00
|
|
|
unwind_plan.AppendRow (row);
|
2011-09-15 08:44:34 +08:00
|
|
|
unwind_plan.SetSourceName ("x86_64 at-func-entry default");
|
2011-05-12 02:39:18 +08:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
ABISysV_x86_64::CreateDefaultUnwindPlan (UnwindPlan &unwind_plan)
|
|
|
|
{
|
|
|
|
uint32_t reg_kind = unwind_plan.GetRegisterKind();
|
|
|
|
uint32_t fp_reg_num = LLDB_INVALID_REGNUM;
|
|
|
|
uint32_t sp_reg_num = LLDB_INVALID_REGNUM;
|
|
|
|
uint32_t pc_reg_num = LLDB_INVALID_REGNUM;
|
|
|
|
|
|
|
|
switch (reg_kind)
|
|
|
|
{
|
|
|
|
case eRegisterKindDWARF:
|
|
|
|
case eRegisterKindGCC:
|
|
|
|
fp_reg_num = gcc_dwarf_rbp;
|
|
|
|
sp_reg_num = gcc_dwarf_rsp;
|
|
|
|
pc_reg_num = gcc_dwarf_rip;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eRegisterKindGDB:
|
|
|
|
fp_reg_num = gdb_rbp;
|
|
|
|
sp_reg_num = gdb_rsp;
|
|
|
|
pc_reg_num = gdb_rip;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case eRegisterKindGeneric:
|
|
|
|
fp_reg_num = LLDB_REGNUM_GENERIC_FP;
|
|
|
|
sp_reg_num = LLDB_REGNUM_GENERIC_SP;
|
|
|
|
pc_reg_num = LLDB_REGNUM_GENERIC_PC;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (fp_reg_num == LLDB_INVALID_REGNUM ||
|
|
|
|
sp_reg_num == LLDB_INVALID_REGNUM ||
|
|
|
|
pc_reg_num == LLDB_INVALID_REGNUM)
|
|
|
|
return false;
|
|
|
|
|
2012-07-14 12:52:53 +08:00
|
|
|
UnwindPlan::RowSP row(new UnwindPlan::Row);
|
2011-05-12 02:39:18 +08:00
|
|
|
|
|
|
|
const int32_t ptr_size = 8;
|
2012-07-14 12:52:53 +08:00
|
|
|
row->SetCFARegister (LLDB_REGNUM_GENERIC_FP);
|
|
|
|
row->SetCFAOffset (2 * ptr_size);
|
|
|
|
row->SetOffset (0);
|
2011-05-12 02:39:18 +08:00
|
|
|
|
2012-07-14 12:52:53 +08:00
|
|
|
row->SetRegisterLocationToAtCFAPlusOffset(fp_reg_num, ptr_size * -2, true);
|
|
|
|
row->SetRegisterLocationToAtCFAPlusOffset(pc_reg_num, ptr_size * -1, true);
|
|
|
|
row->SetRegisterLocationToAtCFAPlusOffset(sp_reg_num, ptr_size * 0, true);
|
2011-05-12 02:39:18 +08:00
|
|
|
|
|
|
|
unwind_plan.AppendRow (row);
|
|
|
|
unwind_plan.SetSourceName ("x86_64 default unwind plan");
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
ABISysV_x86_64::RegisterIsVolatile (const RegisterInfo *reg_info)
|
|
|
|
{
|
|
|
|
return !RegisterIsCalleeSaved (reg_info);
|
|
|
|
}
|
|
|
|
|
2011-08-22 10:49:39 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
2011-05-12 02:39:18 +08:00
|
|
|
bool
|
|
|
|
ABISysV_x86_64::RegisterIsCalleeSaved (const RegisterInfo *reg_info)
|
|
|
|
{
|
|
|
|
if (reg_info)
|
|
|
|
{
|
|
|
|
// Volatile registers include: rbx, rbp, rsp, r12, r13, r14, r15, rip
|
|
|
|
const char *name = reg_info->name;
|
|
|
|
if (name[0] == 'r')
|
|
|
|
{
|
|
|
|
switch (name[1])
|
|
|
|
{
|
|
|
|
case '1': // r12, r13, r14, r15
|
|
|
|
if (name[2] >= '2' && name[2] <= '5')
|
|
|
|
return name[3] == '\0';
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'b': // rbp, rbx
|
|
|
|
if (name[2] == 'p' || name[2] == 'x')
|
|
|
|
return name[3] == '\0';
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 'i': // rip
|
|
|
|
if (name[2] == 'p')
|
|
|
|
return name[3] == '\0';
|
|
|
|
break;
|
|
|
|
|
|
|
|
case 's': // rsp
|
|
|
|
if (name[2] == 'p')
|
|
|
|
return name[3] == '\0';
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2010-06-09 00:52:24 +08:00
|
|
|
void
|
|
|
|
ABISysV_x86_64::Initialize()
|
|
|
|
{
|
|
|
|
PluginManager::RegisterPlugin (pluginName,
|
|
|
|
pluginDesc,
|
|
|
|
CreateInstance);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
ABISysV_x86_64::Terminate()
|
|
|
|
{
|
|
|
|
PluginManager::UnregisterPlugin (CreateInstance);
|
|
|
|
}
|
|
|
|
|
|
|
|
//------------------------------------------------------------------
|
|
|
|
// PluginInterface protocol
|
|
|
|
//------------------------------------------------------------------
|
|
|
|
const char *
|
|
|
|
ABISysV_x86_64::GetPluginName()
|
|
|
|
{
|
|
|
|
return pluginName;
|
|
|
|
}
|
|
|
|
|
|
|
|
const char *
|
|
|
|
ABISysV_x86_64::GetShortPluginName()
|
|
|
|
{
|
|
|
|
return pluginShort;
|
|
|
|
}
|
|
|
|
|
|
|
|
uint32_t
|
|
|
|
ABISysV_x86_64::GetPluginVersion()
|
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|