Initial commit of StackwalkerAPI
[dyninst.git] / stackwalk / src / linux-x86-swk.C
1 /*
2  * Copyright (c) 1996-2007 Barton P. Miller
3  * 
4  * We provide the Paradyn Parallel Performance Tools (below
5  * described as "Paradyn") on an AS IS basis, and do not warrant its
6  * validity or performance.  We reserve the right to update, modify,
7  * or discontinue this software at any time.  We shall have no
8  * obligation to supply such updates or modifications or any other
9  * form of support to you.
10  * 
11  * By your use of Paradyn, you understand and agree that we (or any
12  * other person or entity with proprietary rights in Paradyn) are
13  * under no obligation to provide either maintenance services,
14  * update services, notices of latent defects, or correction of
15  * defects for Paradyn.
16  * 
17  * This library is free software; you can redistribute it and/or
18  * modify it under the terms of the GNU Lesser General Public
19  * License as published by the Free Software Foundation; either
20  * version 2.1 of the License, or (at your option) any later version.
21  * 
22  * This library is distributed in the hope that it will be useful,
23  * but WITHOUT ANY WARRANTY; without even the implied warranty of
24  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
25  * Lesser General Public License for more details.
26  * 
27  * You should have received a copy of the GNU Lesser General Public
28  * License along with this library; if not, write to the Free Software
29  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301  USA
30  */
31
32 #include "stackwalk/h/swk_errors.h"
33 #include "stackwalk/h/walker.h"
34 #include "stackwalk/h/procstate.h"
35 #include "stackwalk/h/framestepper.h"
36 #include "stackwalk/h/basetypes.h"
37 #include "stackwalk/h/frame.h"
38
39 #include "stackwalk/src/linux-swk.h"
40
41 #include "symtabAPI/h/Symtab.h"
42
43 #include <sys/user.h>
44 #include <sys/ptrace.h>
45 #include <string.h>
46 #include <errno.h>
47
48 using namespace Dyninst;
49 using namespace Dyninst::Stackwalker;
50 using namespace Dyninst::SymtabAPI;
51
52 #if defined(arch_x86_64)
53
54 #define PTRACE_REG_IP rip
55 #define PTRACE_REG_FP rbp
56 #define PTRACE_REG_SP rsp
57
58 #include "symtabAPI/h/Symtab.h"
59
60 unsigned ProcDebugLinux::getAddressWidth()
61 {
62    if (cached_addr_width)
63       return cached_addr_width;
64
65    sw_printf("[%s:%u] - Computing address size of proc %d\n",
66              __FILE__, __LINE__, pid);
67    char procname[64];
68    snprintf(procname, 64, "/proc/%d/exe", pid);
69    std::string str(procname);
70    
71    Symtab *obj;
72    bool result = Symtab::openFile(str, obj);
73    if (!result) {
74       sw_printf("[%s:%u] - Error. Failed to open /proc/pid/exe\n");
75       cached_addr_width = sizeof(void*); //Assume 64-bit process
76       return cached_addr_width;
77    }
78
79    sw_printf("[%s:%u] - AMD64 process has address width %d\n",
80              __FILE__, __LINE__, cached_addr_width);
81    cached_addr_width = obj->getAddressWidth();
82    obj->closeFile();
83    
84    return cached_addr_width;
85    return sizeof(void *);
86 }
87
88 #else
89
90 unsigned ProcDebugLinux::getAddressWidth() 
91 {
92    return sizeof(void*);
93 }
94
95 #define PTRACE_REG_IP eip
96 #define PTRACE_REG_FP ebp
97 #define PTRACE_REG_SP esp
98
99 #endif
100
101
102 bool ProcDebugLinux::getRegValue(reg_t reg, THR_ID t, regval_t &val)
103 {
104    user_regs_struct gprs;
105    int result;
106
107    result = ptrace(PTRACE_GETREGS, (pid_t) t, NULL, &gprs);
108    if (result != 0)
109    {
110       int errnum = errno;
111       sw_printf("[%s:%u] - Could not read gprs in %d: %s\n", 
112                 __FILE__, __LINE__, t, strerror(errnum));
113       setLastError(err_procread, "Could not read registers from process");
114       return false;
115    }
116    
117    switch (reg)
118    {
119       case REG_PC:
120          val = gprs.PTRACE_REG_IP;
121          break;
122       case REG_SP:
123          val = gprs.PTRACE_REG_SP;
124          break;
125       case REG_FP:
126          val = gprs.PTRACE_REG_FP;
127          break;
128       default:
129          sw_printf("[%s:%u] - Request for unsupported register %d\n",
130                    __FILE__, __LINE__, reg);
131          setLastError(err_badparam, "Unknown register passed in reg field");
132          return false;
133    }
134    return true;
135 }
136
137 bool Walker::createDefaultSteppers()
138 {
139   FrameStepper *stepper;
140   bool result;
141
142   stepper = new FrameFuncStepper(this);
143   result = addStepper(stepper);
144   if (!result) {
145     sw_printf("[%s:%u] - Error adding stepper %p\n", __FILE__, __LINE__,
146               stepper);
147     return false;
148   }
149
150   return true;
151 }