• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 /**
2  * Copyright (c) 2021-2022 Huawei Device Co., Ltd.
3  * Licensed under the Apache License, Version 2.0 (the "License");
4  * you may not use this file except in compliance with the License.
5  * You may obtain a copy of the License at
6  *
7  * http://www.apache.org/licenses/LICENSE-2.0
8  *
9  * Unless required by applicable law or agreed to in writing, software
10  * distributed under the License is distributed on an "AS IS" BASIS,
11  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12  * See the License for the specific language governing permissions and
13  * limitations under the License.
14  */
15 #include <fstream>
16 #include <sstream>
17 #include <iomanip>
18 #include <memory>
19 #include <algorithm>
20 #include <unwind.h>
21 
22 #include "stacktrace.h"
23 #include "os/mutex.h"
24 
25 #include <cxxabi.h>
26 #include <dlfcn.h>
27 #include "debug_info.h"
28 
29 namespace panda {
30 
31 struct VmaEntry {
32     enum DebugInfoStatus { NOT_READ, VALID, BAD };
33 
34     // NOLINTNEXTLINE(modernize-pass-by-value)
VmaEntrypanda::VmaEntry35     VmaEntry(uintptr_t param_start_addr, uintptr_t param_end_addr, uintptr_t param_offset, const std::string &fname)
36         : start_addr(param_start_addr), end_addr(param_end_addr), offset(param_offset), filename(fname)
37     {
38     }
39 
40     ~VmaEntry() = default;
41 
42     uintptr_t start_addr;               // NOLINT(misc-non-private-member-variables-in-classes)
43     uintptr_t end_addr;                 // NOLINT(misc-non-private-member-variables-in-classes)
44     uintptr_t offset;                   // NOLINT(misc-non-private-member-variables-in-classes)
45     std::string filename;               // NOLINT(misc-non-private-member-variables-in-classes)
46     DebugInfoStatus status {NOT_READ};  // NOLINT(misc-non-private-member-variables-in-classes)
47     DebugInfo debug_info;               // NOLINT(misc-non-private-member-variables-in-classes)
48 
49     DEFAULT_MOVE_SEMANTIC(VmaEntry);
50     NO_COPY_SEMANTIC(VmaEntry);
51 };
52 
53 class Tokenizer {
54 public:
55     // NOLINTNEXTLINE(modernize-pass-by-value)
Tokenizer(const std::string & str)56     explicit Tokenizer(const std::string &str) : str_(str), pos_(0) {}
57 
Next(char delim=' ')58     std::string Next(char delim = ' ')
59     {
60         while (pos_ < str_.length() && str_[pos_] == ' ') {
61             ++pos_;
62         }
63         size_t pos = str_.find(delim, pos_);
64         std::string token;
65         if (pos == std::string::npos) {
66             token = str_.substr(pos_);
67             pos_ = str_.length();
68         } else {
69             token = str_.substr(pos_, pos - pos_);
70             pos_ = pos + 1;  // skip delimiter
71         }
72         return token;
73     }
74 
75 private:
76     std::string str_;
77     size_t pos_;
78 };
79 
80 class StackPrinter {
81 public:
GetInstance()82     static StackPrinter &GetInstance()
83     {
84         static StackPrinter printer;
85         return printer;
86     }
87 
Print(const std::vector<uintptr_t> & stacktrace,std::ostream & out)88     std::ostream &Print(const std::vector<uintptr_t> &stacktrace, std::ostream &out)
89     {
90         os::memory::LockHolder lock(mutex_);
91         ScanVma();
92         for (size_t frame_num = 0; frame_num < stacktrace.size(); ++frame_num) {
93             PrintFrame(frame_num, stacktrace[frame_num], out);
94         }
95         return out;
96     }
97 
98     NO_MOVE_SEMANTIC(StackPrinter);
99     NO_COPY_SEMANTIC(StackPrinter);
100 
101 private:
102     explicit StackPrinter() = default;
103     ~StackPrinter() = default;
104 
PrintFrame(size_t frame_num,uintptr_t pc,std::ostream & out)105     void PrintFrame(size_t frame_num, uintptr_t pc, std::ostream &out)
106     {
107         std::ios_base::fmtflags f = out.flags();
108         auto w = out.width();
109         out << "#" << std::setw(2U) << std::left << frame_num << ": 0x" << std::hex << pc << " ";
110         out.flags(f);
111         out.width(w);
112 
113         VmaEntry *vma = FindVma(pc);
114         if (vma == nullptr) {
115             vmas_.clear();
116             ScanVma();
117             vma = FindVma(pc);
118         }
119         if (vma != nullptr) {
120             uintptr_t pc_offset = pc - vma->start_addr + vma->offset;
121             // pc points to the instruction after the call
122             // Decrement pc to get source line number pointing to the function call
123             --pc_offset;
124             std::string function;
125             std::string src_file;
126             unsigned int line = 0;
127             if (ReadDebugInfo(vma) && vma->debug_info.GetSrcLocation(pc_offset, &function, &src_file, &line)) {
128                 PrintFrame(function, src_file, line, out);
129                 return;
130             }
131             uintptr_t offset = 0;
132             if (ReadSymbol(pc, &function, &offset)) {
133                 PrintFrame(function, offset, out);
134                 return;
135             }
136         }
137         out << "??:??\n";
138     }
139 
PrintFrame(const std::string & function,const std::string & src_file,unsigned int line,std::ostream & out)140     void PrintFrame(const std::string &function, const std::string &src_file, unsigned int line, std::ostream &out)
141     {
142         if (function.empty()) {
143             out << "??";
144         } else {
145             Demangle(function, out);
146         }
147         out << "\n     at ";
148         if (src_file.empty()) {
149             out << "??";
150         } else {
151             out << src_file;
152         }
153         out << ":";
154         if (line == 0) {
155             out << "??";
156         } else {
157             out << line;
158         }
159 
160         out << "\n";
161     }
162 
PrintFrame(const std::string & function,uintptr_t offset,std::ostream & out)163     void PrintFrame(const std::string &function, uintptr_t offset, std::ostream &out)
164     {
165         std::ios_base::fmtflags f = out.flags();
166         Demangle(function, out);
167         out << std::hex << "+0x" << offset << "\n";
168         out.flags(f);
169     }
170 
ReadSymbol(uintptr_t pc,std::string * function,uintptr_t * offset)171     bool ReadSymbol(uintptr_t pc, std::string *function, uintptr_t *offset)
172     {
173         Dl_info info {};
174         if (dladdr(reinterpret_cast<void *>(pc), &info) != 0 && info.dli_sname != nullptr) {
175             *function = info.dli_sname;
176             *offset = pc - reinterpret_cast<uintptr_t>(info.dli_saddr);
177             return true;
178         }
179         return false;
180     }
181 
Demangle(const std::string & function,std::ostream & out)182     void Demangle(const std::string &function, std::ostream &out)
183     {
184         size_t length = 0;
185         int status = 0;
186         char *demangled_function = abi::__cxa_demangle(function.c_str(), nullptr, &length, &status);
187         if (status == 0) {
188             out << demangled_function;
189             free(demangled_function);  // NOLINT(cppcoreguidelines-no-malloc)
190         } else {
191             out << function;
192         }
193     }
194 
FindVma(uintptr_t pc)195     VmaEntry *FindVma(uintptr_t pc)
196     {
197         VmaEntry el(pc, pc, 0, "");
198         auto it = std::upper_bound(vmas_.begin(), vmas_.end(), el,
199                                    [](const VmaEntry &e1, const VmaEntry &e2) { return e1.end_addr < e2.end_addr; });
200         if (it != vmas_.end() && (it->start_addr <= pc && pc < it->end_addr)) {
201             return &(*it);
202         }
203         return nullptr;
204     }
205 
ReadDebugInfo(VmaEntry * vma)206     bool ReadDebugInfo(VmaEntry *vma)
207     {
208         if (vma->status == VmaEntry::VALID) {
209             return true;
210         }
211         if (vma->status == VmaEntry::BAD) {
212             return false;
213         }
214         if (!vma->filename.empty() && vma->debug_info.ReadFromFile(vma->filename.c_str()) == DebugInfo::SUCCESS) {
215             vma->status = VmaEntry::VALID;
216             return true;
217         }
218         vma->status = VmaEntry::BAD;
219         return false;
220     }
221 
ScanVma()222     void ScanVma()
223     {
224         static const int HEX_RADIX = 16;
225         static const size_t MODE_FIELD_LEN = 4;
226         static const size_t XMODE_POS = 2;
227 
228         if (!vmas_.empty()) {
229             return;
230         }
231 
232         std::stringstream fname;
233         fname << "/proc/self/maps";
234         std::string filename = fname.str();
235         std::ifstream maps(filename.c_str());
236 
237         while (maps) {
238             std::string line;
239             std::getline(maps, line);
240             Tokenizer tokenizer(line);
241             std::string start_addr = tokenizer.Next('-');
242             std::string end_addr = tokenizer.Next();
243             std::string rights = tokenizer.Next();
244             if (rights.length() == MODE_FIELD_LEN && rights[XMODE_POS] == 'x') {
245                 std::string offset = tokenizer.Next();
246                 tokenizer.Next();
247                 tokenizer.Next();
248                 std::string obj_filename = tokenizer.Next();
249                 vmas_.emplace_back(stoul(start_addr, nullptr, HEX_RADIX), stoul(end_addr, nullptr, HEX_RADIX),
250                                    stoul(offset, nullptr, HEX_RADIX), obj_filename);
251             }
252         }
253     }
254 
255 private:
256     std::vector<VmaEntry> vmas_;
257     os::memory::Mutex mutex_;
258 };
259 
260 class Buf {
261 public:
Buf(uintptr_t * buf,size_t skip,size_t capacity)262     Buf(uintptr_t *buf, size_t skip, size_t capacity) : buf_(buf), skip_(skip), size_(0), capacity_(capacity) {}
263 
Append(uintptr_t pc)264     void Append(uintptr_t pc)
265     {
266         if (skip_ > 0) {
267             // Skip the element
268             --skip_;
269             return;
270         }
271         if (size_ >= capacity_) {
272             return;
273         }
274         buf_[size_++] = pc;  // NOLINT(cppcoreguidelines-pro-bounds-pointer-arithmetic)
275     }
276 
Size() const277     int Size() const
278     {
279         return size_;
280     }
281 
282 private:
283     uintptr_t *buf_;
284     size_t skip_;
285     size_t size_;
286     size_t capacity_;
287 };
288 
FrameHandler(struct _Unwind_Context * ctx,void * arg)289 static _Unwind_Reason_Code FrameHandler(struct _Unwind_Context *ctx, [[maybe_unused]] void *arg)
290 {
291     Buf *buf = reinterpret_cast<Buf *>(arg);
292     uintptr_t pc = _Unwind_GetIP(ctx);
293     // _Unwind_GetIP returns 0 pc at the end of the stack. Ignore it
294     if (pc != 0) {
295         buf->Append(pc);
296     }
297     return _URC_NO_REASON;
298 }
299 
GetStacktrace()300 std::vector<uintptr_t> GetStacktrace()
301 {
302     static constexpr size_t BUF_SIZE = 100;
303     static constexpr int SKIP_FRAMES = 2;  // backtrace
304     std::vector<uintptr_t> buf;
305     buf.resize(BUF_SIZE);
306     Buf buf_wrapper(buf.data(), SKIP_FRAMES, buf.size());
307     _Unwind_Reason_Code res = _Unwind_Backtrace(FrameHandler, &buf_wrapper);
308     if (res != _URC_END_OF_STACK || buf_wrapper.Size() < 0) {
309         return std::vector<uintptr_t>();
310     }
311 
312     buf.resize(buf_wrapper.Size());
313     return buf;
314 }
315 
PrintStack(const std::vector<uintptr_t> & stacktrace,std::ostream & out)316 std::ostream &PrintStack(const std::vector<uintptr_t> &stacktrace, std::ostream &out)
317 {
318     return StackPrinter::GetInstance().Print(stacktrace, out);
319 }
320 
321 }  // namespace panda
322