GCC 4.8 build fixes: ensure all extern template declarations are in fact extern'ed...
[dyninst.git] / stackwalk / src / freebsd-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 sp_offset_32 = 120;
57 static const int fp_offset_32 = 76;
58 static const int pc_offset_32 = 108;
59 static const int sp_offset_64 = 216;
60 static const int fp_offset_64 = 104;
61 static const int pc_offset_64 = 192;
62
63 gcframe_ret_t SigHandlerStepperImpl::getCallerFrame(const Frame &in, Frame &out)
64 {
65    int fp_offset;
66    int pc_offset;
67    int sp_offset;
68    bool result;
69    int addr_size = getProcessState()->getAddressWidth();
70    if (addr_size == 4) {
71       fp_offset = fp_offset_32;
72       pc_offset = pc_offset_32;
73       sp_offset = sp_offset_32;
74    }
75    else {
76       fp_offset = fp_offset_64;
77       pc_offset = pc_offset_64;
78       sp_offset = sp_offset_64;
79   }
80
81    location_t fp_loc;
82    Address fp = 0x0;
83    fp_loc.location = loc_address;
84    fp_loc.val.addr = in.getSP() + fp_offset;
85    sw_printf("[%s:%u] - SigHandler Reading FP from %lx\n",
86              FILE__, __LINE__, fp_loc.val.addr);
87    result = getProcessState()->readMem(&fp, fp_loc.val.addr, addr_size);
88    if (!result) {
89       return gcf_error;
90    }
91
92    location_t pc_loc;
93    Address pc = 0x0;
94    pc_loc.location = loc_address;
95    pc_loc.val.addr = in.getSP() + pc_offset;
96    sw_printf("[%s:%u] - SigHandler Reading PC from %lx\n",
97              FILE__, __LINE__, pc_loc.val.addr);
98    result = getProcessState()->readMem(&pc, pc_loc.val.addr, addr_size);
99    if (!result) {
100       return gcf_error;
101    }
102
103    location_t sp_loc;
104    Address sp = 0x0;
105    sp_loc.location = loc_address;
106    sp_loc.val.addr = in.getSP() + sp_offset;
107    sw_printf("[%s:%u] - SigHandler Reading PC from %lx\n",
108              FILE__, __LINE__, sp_loc.val.addr);
109    result = getProcessState()->readMem(&sp, sp_loc.val.addr, addr_size);
110    if (!result) {
111       return gcf_error;
112    }    
113
114    out.setRA((Dyninst::MachRegisterVal) pc);
115    out.setFP((Dyninst::MachRegisterVal) fp);
116    out.setSP((Dyninst::MachRegisterVal) sp);
117    out.setRALocation(pc_loc);
118    out.setFPLocation(fp_loc);
119    out.setSPLocation(sp_loc);
120    out.setNonCall();
121    return gcf_success;
122 }
123
124 void SigHandlerStepperImpl::registerStepperGroup(StepperGroup *group)
125 {
126    int addr_size = getProcessState()->getAddressWidth();
127
128    // FreeBSD signal handler tramps are in a static location
129    // at the top of the user space stack
130    if (addr_size == 4) {
131      group->addStepper(parent_stepper, 0xbfbfffb4, 0xbfbfffb4+1);
132    }
133 #if defined(arch_64bit)
134    else {
135      group->addStepper(parent_stepper, 0x00007fffffffffc4, 0x00007fffffffffc4+1);
136    }  
137 #endif
138 }