Merge branch 'v8.2' of ssh://git.dyninst.org/pub/dyninst into v8.2
[dyninst.git] / stackwalk / src / win-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 "SymtabReader.h"
32 #include "stackwalk/h/walker.h"
33 #include "stackwalk/h/framestepper.h"
34 #include "stackwalk/h/steppergroup.h"
35 #include "stackwalk/h/swk_errors.h"
36 #include "common/src/ntHeaders.h"
37
38 #include <windows.h>
39
40
41 #include "stackwalk/h/procstate.h"
42 #include "stackwalk/h/frame.h"
43
44 #include "stackwalk/src/sw.h"
45
46 #include <assert.h>
47
48
49 using namespace Dyninst;
50 using namespace Dyninst::Stackwalker;
51
52 bool Walker::createDefaultSteppers()
53 {
54   FrameStepper *stepper;
55   bool result = true;
56
57   stepper = new FrameFuncStepper(this);
58   result = addStepper(stepper);
59   if (!result)
60      goto error;
61   sw_printf("[%s:%u] - Stepper %p is FrameFuncStepper\n",
62             FILE__, __LINE__, stepper);
63 #if defined(USE_PARSE_API)
64   stepper = new AnalysisStepper(this);
65   result = addStepper(stepper);
66   if (!result)
67                 goto error;
68 #endif
69
70   return true;
71  error:
72   sw_printf("[%s:%u] - Error adding stepper %p\n", stepper);
73   return false;
74 }
75
76 bool ProcSelf::getThreadIds(std::vector<THR_ID> &threads)
77 {
78   bool result;
79   THR_ID tid;
80
81   result = getDefaultThread(tid);
82   if (!result) {
83     sw_printf("[%s:%u] - Could not read default thread\n",
84                FILE__, __LINE__);
85     return false;
86   }
87   threads.clear();
88   threads.push_back(tid);
89   return true;
90 }
91
92 bool ProcSelf::getDefaultThread(THR_ID &default_tid)
93 {
94   default_tid = GetCurrentThread();
95   return true;
96 }
97
98 bool ProcSelf::readMem(void *dest, Address source, size_t size)
99 {
100    memcpy(dest, (const void *) source, size);
101    return true;
102 }
103
104 ProcSelf::ProcSelf(std::string exe_path) :
105    ProcessState(P_getpid(), exe_path)
106 {
107 }
108
109 void ProcSelf::initialize()
110 {
111 }
112
113 bool LibraryState::updateLibsArch(std::vector<std::pair<LibAddrPair, unsigned int> > &alibs)
114 {
115         return true;
116 }
117
118 SymbolReaderFactory* Stackwalker::getDefaultSymbolReader()
119 {
120         if (NULL == Walker::getSymbolReader()) {
121                 static SymtabAPI::SymtabReaderFactory fact;
122                 Walker::setSymbolReader(&fact);
123         }
124         return Walker::getSymbolReader();
125 }
126
127 static const char* START_FUNC_NAME = "_start";
128 static const char* CLONE_FUNC_NAME = "clone";
129 //static const char* START_THREAD_FUNC_NAME = 
130
131 void BottomOfStackStepperImpl::initialize()
132 {
133         // For now, we stop when we get a return address of 0
134         ra_stack_tops.push_back(std::pair<Address, Address>(0, 0));
135         // By examination, this is an _start equivalent
136         ra_stack_tops.push_back(std::pair<Address, Address>(0x77959ECB, 0x77959F01));
137         ra_stack_tops.push_back(std::pair<Address, Address>(0x77959EAA, 0x77959EC6));
138 /*   ProcessState *proc = walker->getProcessState();
139    assert(proc);
140
141    sw_printf("[%s:%u] - Initializing BottomOfStackStepper\n", FILE__, __LINE__);
142    
143    LibraryState *libs = proc->getLibraryTracker();
144    if (!libs) {
145       sw_printf("[%s:%u] - Error initing StackBottom.  No library state for process.\n",
146                 FILE__, __LINE__);
147       return;
148    }
149    SymbolReaderFactory *fact = Walker::getSymbolReader();
150    if (!fact) {
151       sw_printf("[%s:%u] - Failed to get symbol reader\n");
152       return;
153    }
154
155    if (!aout_init)
156    {
157       LibAddrPair aout_addr;
158       SymReader *aout = NULL;
159       Symbol_t start_sym;
160       bool result = libs->getAOut(aout_addr);
161       if (result) {
162          aout = fact->openSymbolReader(aout_addr.first);
163          aout_init = true;
164       }
165       if (aout) {
166          start_sym = aout->getSymbolByName(START_FUNC_NAME);
167          if (aout->isValidSymbol(start_sym)) {
168             Dyninst::Address start = aout->getSymbolOffset(start_sym)+aout_addr.second;
169             Dyninst::Address end = aout->getSymbolSize(start_sym) + start;
170             if (start == end) {
171                sw_printf("[%s:%u] - %s symbol has 0 length, using length of %lu\n",
172                        FILE__, __LINE__, START_FUNC_NAME, START_HEURISTIC_LENGTH);
173                end = start + START_HEURISTIC_LENGTH;
174             }
175             sw_printf("[%s:%u] - Bottom stepper taking %lx to %lx for start\n", 
176                       FILE__, __LINE__, start, end);
177             ra_stack_tops.push_back(std::pair<Address, Address>(start, end));
178          }
179       }
180    }
181
182    if (!libthread_init)
183    {
184       LibAddrPair libthread_addr;
185       SymReader *libthread = NULL;
186       Symbol_t clone_sym, startthread_sym;
187       bool result = libs->getLibthread(libthread_addr);
188       if (result) {
189          libthread = fact->openSymbolReader(libthread_addr.first);
190          libthread_init = true;
191       }
192       if (libthread) {
193          clone_sym = libthread->getSymbolByName(CLONE_FUNC_NAME);
194          if (libthread->isValidSymbol(clone_sym)) {
195             Dyninst::Address start = libthread->getSymbolOffset(clone_sym) + 
196                libthread_addr.second;
197             Dyninst::Address end = libthread->getSymbolSize(clone_sym) + start;
198             sw_printf("[%s:%u] - Bottom stepper taking %lx to %lx for clone\n", 
199                       FILE__, __LINE__, start, end);
200             ra_stack_tops.push_back(std::pair<Address, Address>(start, end));
201          }
202          startthread_sym = libthread->getSymbolByName(START_THREAD_FUNC_NAME);
203          if (libthread->isValidSymbol(startthread_sym)) {
204             Dyninst::Address start = libthread->getSymbolOffset(startthread_sym) + 
205                libthread_addr.second;
206             Dyninst::Address end = libthread->getSymbolSize(startthread_sym) + start;
207             sw_printf("[%s:%u] - Bottom stepper taking %lx to %lx for start_thread\n", 
208                       FILE__, __LINE__, start, end);
209             ra_stack_tops.push_back(std::pair<Address, Address>(start, end));
210          }
211       }
212    }
213 */
214 }
215 void BottomOfStackStepperImpl::newLibraryNotification(LibAddrPair *, lib_change_t)
216 {
217
218 }