Initialize thrd in a new FrameNode ctor
[dyninst.git] / stackwalk / src / linux-x86-swk.C
1 /*
2  * See the dyninst/COPYRIGHT file for copyright information.
3  * 
4  * We provide the Paradyn Tools (below described as "Paradyn")
5  * on an AS IS basis, and do not warrant its validity or performance.
6  * We reserve the right to update, modify, or discontinue this
7  * software at any time.  We shall have no obligation to supply such
8  * updates or modifications or any other form of support to you.
9  * 
10  * By your use of Paradyn, you understand and agree that we (or any
11  * other person or entity with proprietary rights in Paradyn) are
12  * under no obligation to provide either maintenance services,
13  * update services, notices of latent defects, or correction of
14  * defects for Paradyn.
15  * 
16  * This library is free software; you can redistribute it and/or
17  * modify it under the terms of the GNU Lesser General Public
18  * License as published by the Free Software Foundation; either
19  * version 2.1 of the License, or (at your option) any later version.
20  * 
21  * This library is distributed in the hope that it will be useful,
22  * but WITHOUT ANY WARRANTY; without even the implied warranty of
23  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
24  * Lesser General Public License for more details.
25  * 
26  * You should have received a copy of the GNU Lesser General Public
27  * License along with this library; if not, write to the Free Software
28  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
29  */
30
31 #include "stackwalk/h/swk_errors.h"
32 #include "stackwalk/h/walker.h"
33 #include "stackwalk/h/procstate.h"
34 #include "stackwalk/h/framestepper.h"
35 #include "stackwalk/h/basetypes.h"
36 #include "stackwalk/h/frame.h"
37
38 #include "stackwalk/src/symtab-swk.h"
39 #include "stackwalk/src/linuxbsd-swk.h"
40 #include "stackwalk/src/dbgstepper-impl.h"
41 #include "stackwalk/src/x86-swk.h"
42
43 #include "common/src/Types.h"
44
45 #include <sys/user.h>
46 #include <sys/ptrace.h>
47 #include <sys/types.h>
48 #include <sys/stat.h>
49 #include <fcntl.h>
50 #include <string.h>
51 #include <errno.h>
52
53 using namespace Dyninst;
54 using namespace Dyninst::Stackwalker;
55
56 static const int fp_offset_32 = 28;
57 static const int pc_offset_32 = 60;
58 static const int frame_size_32 = 736;
59 static const int fp_offset_64 = 120;
60 static const int pc_offset_64 = 168;
61 static const int frame_size_64 = 1088;
62
63 struct sigframe_offsets {
64    int fp_offset;
65    int pc_offset;
66    int sp_offset;
67    int frame_size;
68 };
69
70 int frames_32_size = 3;
71 sigframe_offsets frames_32[] = { {28, 60, 32, 728}, {28, 60, 32, 736}, 
72                                  {28, 60, 32, 0} };
73
74 int frames_64_size = 6;
75 sigframe_offsets frames_64[] = { {120, 168, 160, 568}, {120, 168, 160, 1096}, 
76                                  {120, 168, 160, 1112}, {120, 168, 160, 1180},
77                                  {120, 168, 160, 1120}, {120, 168, 160, 0} };
78
79
80 gcframe_ret_t SigHandlerStepperImpl::getCallerFrame(const Frame &in, Frame &out)
81 {
82    bool result;
83    int addr_size = getProcessState()->getAddressWidth();
84
85    unsigned frames_size;
86    sigframe_offsets *frames;
87    if (addr_size == 4) {
88       frames_size = frames_32_size;
89       frames = frames_32;
90    }
91    else {
92       frames_size = frames_64_size;
93       frames = frames_64;
94    }
95
96    Address last_read_sp_addr = 0;
97    Address last_read_sp_val = 0;
98
99    for (unsigned i = 0; i < frames_size; i++) {
100       location_t sp_loc;
101       sp_loc.location = loc_address;
102       sp_loc.val.addr = in.getSP() + frames[i].sp_offset;
103       Address sp = 0;
104       if (last_read_sp_addr != sp_loc.val.addr)
105       {
106          result = getProcessState()->readMem(&sp, sp_loc.val.addr, addr_size);
107          if (!result) {
108             sw_printf("[%s:%u] Unexpected error reading from stack memory 0x%lx for signal frame\n",
109                       __FILE__, __LINE__);
110             return gcf_error;
111          }
112          last_read_sp_addr = sp_loc.val.addr;
113          last_read_sp_val = sp;
114       }
115       else {
116          sp = last_read_sp_val;
117       }
118
119       if (frames[i].frame_size && (sp != in.getSP() + frames[i].frame_size)) {
120          sw_printf("[%s:%u] - Signal frame candidate %d does not fit (%lx != %lx). Trying another.\n",
121                    __FILE__, __LINE__, i, sp, in.getSP() + frames[i].frame_size);
122          continue;
123       }
124       sw_printf("[%s:%u] - Using signal frame candidate %d\n", __FILE__, __LINE__, i);
125
126       location_t fp_loc;
127       Address fp = 0x0;
128       fp_loc.location = loc_address;
129       fp_loc.val.addr = in.getSP() + frames[i].fp_offset;
130       sw_printf("[%s:%u] - SigHandler Reading FP from %lx\n",
131                 __FILE__, __LINE__, fp_loc.val.addr);
132       result = getProcessState()->readMem(&fp, fp_loc.val.addr, addr_size);
133       if (!result) {
134          sw_printf("[%s:%u] Unexpected error reading from stack memory 0x%lx for signal frame\n",
135                    __FILE__, __LINE__);
136          return gcf_error;
137       }
138
139       location_t pc_loc;
140       Address pc = 0x0;
141       pc_loc.location = loc_address;
142       pc_loc.val.addr = in.getSP() + frames[i].pc_offset;
143       sw_printf("[%s:%u] - SigHandler Reading PC from %lx\n",
144                 __FILE__, __LINE__, pc_loc.val.addr);
145       result = getProcessState()->readMem(&pc, pc_loc.val.addr, addr_size);
146       if (!result) {
147          sw_printf("[%s:%u] Unexpected error reading from stack memory 0x%lx for signal frame\n",
148                    __FILE__, __LINE__);
149          return gcf_error;
150       }
151
152       out.setRA((Dyninst::MachRegisterVal) pc);
153       out.setFP((Dyninst::MachRegisterVal) fp);
154       out.setSP((Dyninst::MachRegisterVal) sp);
155       out.setRALocation(pc_loc);
156       out.setFPLocation(fp_loc);
157       out.setSPLocation(sp_loc);
158       out.setNonCall();
159       return gcf_success;
160    }
161    
162    sw_printf("[%s:%u] - Could not find matching candidate for signal frame\n", __FILE__, __LINE__);
163    return gcf_not_me;
164 }