1 //===-- StopInfoMachException.cpp -----------------------------------------===//
2 //
3 // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
4 // See https://llvm.org/LICENSE.txt for license information.
5 // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
6 //
7 //===----------------------------------------------------------------------===//
8 
9 #include "StopInfoMachException.h"
10 
11 #include "lldb/lldb-forward.h"
12 
13 #if defined(__APPLE__)
14 // Needed for the EXC_RESOURCE interpretation macros
15 #include <kern/exc_resource.h>
16 #endif
17 
18 #include "lldb/Breakpoint/Watchpoint.h"
19 #include "lldb/Symbol/Symbol.h"
20 #include "lldb/Target/DynamicLoader.h"
21 #include "lldb/Target/ExecutionContext.h"
22 #include "lldb/Target/Process.h"
23 #include "lldb/Target/RegisterContext.h"
24 #include "lldb/Target/Target.h"
25 #include "lldb/Target/Thread.h"
26 #include "lldb/Target/ThreadPlan.h"
27 #include "lldb/Target/UnixSignals.h"
28 #include "lldb/Utility/StreamString.h"
29 
30 using namespace lldb;
31 using namespace lldb_private;
32 
GetDescription()33 const char *StopInfoMachException::GetDescription() {
34   if (!m_description.empty())
35     return m_description.c_str();
36   if (GetValue() == eStopReasonInvalid)
37     return "invalid stop reason!";
38 
39   ExecutionContext exe_ctx(m_thread_wp.lock());
40   Target *target = exe_ctx.GetTargetPtr();
41   const llvm::Triple::ArchType cpu =
42       target ? target->GetArchitecture().GetMachine()
43              : llvm::Triple::UnknownArch;
44 
45   const char *exc_desc = nullptr;
46   const char *code_label = "code";
47   const char *code_desc = nullptr;
48   const char *subcode_label = "subcode";
49   const char *subcode_desc = nullptr;
50 
51 #if defined(__APPLE__)
52   char code_desc_buf[32];
53   char subcode_desc_buf[32];
54 #endif
55 
56   switch (m_value) {
57   case 1: // EXC_BAD_ACCESS
58     exc_desc = "EXC_BAD_ACCESS";
59     subcode_label = "address";
60     switch (cpu) {
61     case llvm::Triple::x86:
62     case llvm::Triple::x86_64:
63       switch (m_exc_code) {
64       case 0xd:
65         code_desc = "EXC_I386_GPFLT";
66         m_exc_data_count = 1;
67         break;
68       }
69       break;
70     case llvm::Triple::arm:
71     case llvm::Triple::thumb:
72       switch (m_exc_code) {
73       case 0x101:
74         code_desc = "EXC_ARM_DA_ALIGN";
75         break;
76       case 0x102:
77         code_desc = "EXC_ARM_DA_DEBUG";
78         break;
79       }
80       break;
81 
82     default:
83       break;
84     }
85     break;
86 
87   case 2: // EXC_BAD_INSTRUCTION
88     exc_desc = "EXC_BAD_INSTRUCTION";
89     switch (cpu) {
90     case llvm::Triple::x86:
91     case llvm::Triple::x86_64:
92       if (m_exc_code == 1)
93         code_desc = "EXC_I386_INVOP";
94       break;
95 
96     case llvm::Triple::arm:
97     case llvm::Triple::thumb:
98       if (m_exc_code == 1)
99         code_desc = "EXC_ARM_UNDEFINED";
100       break;
101 
102     default:
103       break;
104     }
105     break;
106 
107   case 3: // EXC_ARITHMETIC
108     exc_desc = "EXC_ARITHMETIC";
109     switch (cpu) {
110     case llvm::Triple::x86:
111     case llvm::Triple::x86_64:
112       switch (m_exc_code) {
113       case 1:
114         code_desc = "EXC_I386_DIV";
115         break;
116       case 2:
117         code_desc = "EXC_I386_INTO";
118         break;
119       case 3:
120         code_desc = "EXC_I386_NOEXT";
121         break;
122       case 4:
123         code_desc = "EXC_I386_EXTOVR";
124         break;
125       case 5:
126         code_desc = "EXC_I386_EXTERR";
127         break;
128       case 6:
129         code_desc = "EXC_I386_EMERR";
130         break;
131       case 7:
132         code_desc = "EXC_I386_BOUND";
133         break;
134       case 8:
135         code_desc = "EXC_I386_SSEEXTERR";
136         break;
137       }
138       break;
139 
140     default:
141       break;
142     }
143     break;
144 
145   case 4: // EXC_EMULATION
146     exc_desc = "EXC_EMULATION";
147     break;
148 
149   case 5: // EXC_SOFTWARE
150     exc_desc = "EXC_SOFTWARE";
151     if (m_exc_code == 0x10003) {
152       subcode_desc = "EXC_SOFT_SIGNAL";
153       subcode_label = "signo";
154     }
155     break;
156 
157   case 6: // EXC_BREAKPOINT
158   {
159     exc_desc = "EXC_BREAKPOINT";
160     switch (cpu) {
161     case llvm::Triple::x86:
162     case llvm::Triple::x86_64:
163       switch (m_exc_code) {
164       case 1:
165         code_desc = "EXC_I386_SGL";
166         break;
167       case 2:
168         code_desc = "EXC_I386_BPT";
169         break;
170       }
171       break;
172 
173     case llvm::Triple::arm:
174     case llvm::Triple::thumb:
175       switch (m_exc_code) {
176       case 0x101:
177         code_desc = "EXC_ARM_DA_ALIGN";
178         break;
179       case 0x102:
180         code_desc = "EXC_ARM_DA_DEBUG";
181         break;
182       case 1:
183         code_desc = "EXC_ARM_BREAKPOINT";
184         break;
185       // FIXME temporary workaround, exc_code 0 does not really mean
186       // EXC_ARM_BREAKPOINT
187       case 0:
188         code_desc = "EXC_ARM_BREAKPOINT";
189         break;
190       }
191       break;
192 
193     default:
194       break;
195     }
196   } break;
197 
198   case 7:
199     exc_desc = "EXC_SYSCALL";
200     break;
201 
202   case 8:
203     exc_desc = "EXC_MACH_SYSCALL";
204     break;
205 
206   case 9:
207     exc_desc = "EXC_RPC_ALERT";
208     break;
209 
210   case 10:
211     exc_desc = "EXC_CRASH";
212     break;
213   case 11:
214     exc_desc = "EXC_RESOURCE";
215 #if defined(__APPLE__)
216     {
217       int resource_type = EXC_RESOURCE_DECODE_RESOURCE_TYPE(m_exc_code);
218 
219       code_label = "limit";
220       code_desc = code_desc_buf;
221       subcode_label = "observed";
222       subcode_desc = subcode_desc_buf;
223 
224       switch (resource_type) {
225       case RESOURCE_TYPE_CPU:
226         exc_desc = "EXC_RESOURCE RESOURCE_TYPE_CPU";
227         snprintf(code_desc_buf, sizeof(code_desc_buf), "%d%%",
228                  (int)EXC_RESOURCE_CPUMONITOR_DECODE_PERCENTAGE(m_exc_code));
229         snprintf(subcode_desc_buf, sizeof(subcode_desc_buf), "%d%%",
230                  (int)EXC_RESOURCE_CPUMONITOR_DECODE_PERCENTAGE_OBSERVED(
231                      m_exc_subcode));
232         break;
233       case RESOURCE_TYPE_WAKEUPS:
234         exc_desc = "EXC_RESOURCE RESOURCE_TYPE_WAKEUPS";
235         snprintf(
236             code_desc_buf, sizeof(code_desc_buf), "%d w/s",
237             (int)EXC_RESOURCE_CPUMONITOR_DECODE_WAKEUPS_PERMITTED(m_exc_code));
238         snprintf(subcode_desc_buf, sizeof(subcode_desc_buf), "%d w/s",
239                  (int)EXC_RESOURCE_CPUMONITOR_DECODE_WAKEUPS_OBSERVED(
240                      m_exc_subcode));
241         break;
242       case RESOURCE_TYPE_MEMORY:
243         exc_desc = "EXC_RESOURCE RESOURCE_TYPE_MEMORY";
244         snprintf(code_desc_buf, sizeof(code_desc_buf), "%d MB",
245                  (int)EXC_RESOURCE_HWM_DECODE_LIMIT(m_exc_code));
246         subcode_desc = nullptr;
247         subcode_label = "unused";
248         break;
249 #if defined(RESOURCE_TYPE_IO)
250       // RESOURCE_TYPE_IO is introduced in macOS SDK 10.12.
251       case RESOURCE_TYPE_IO:
252         exc_desc = "EXC_RESOURCE RESOURCE_TYPE_IO";
253         snprintf(code_desc_buf, sizeof(code_desc_buf), "%d MB",
254                  (int)EXC_RESOURCE_IO_DECODE_LIMIT(m_exc_code));
255         snprintf(subcode_desc_buf, sizeof(subcode_desc_buf), "%d MB",
256                  (int)EXC_RESOURCE_IO_OBSERVED(m_exc_subcode));
257         ;
258         break;
259 #endif
260       }
261     }
262 #endif
263     break;
264   case 12:
265     exc_desc = "EXC_GUARD";
266     break;
267   }
268 
269   StreamString strm;
270 
271   if (exc_desc)
272     strm.PutCString(exc_desc);
273   else
274     strm.Printf("EXC_??? (%" PRIu64 ")", m_value);
275 
276   if (m_exc_data_count >= 1) {
277     if (code_desc)
278       strm.Printf(" (%s=%s", code_label, code_desc);
279     else
280       strm.Printf(" (%s=%" PRIu64, code_label, m_exc_code);
281   }
282 
283   if (m_exc_data_count >= 2) {
284     if (subcode_desc)
285       strm.Printf(", %s=%s", subcode_label, subcode_desc);
286     else
287       strm.Printf(", %s=0x%" PRIx64, subcode_label, m_exc_subcode);
288   }
289 
290   if (m_exc_data_count > 0)
291     strm.PutChar(')');
292 
293   m_description = std::string(strm.GetString());
294   return m_description.c_str();
295 }
296 
GetStopInfoForHardwareBP(Thread & thread,Target * target,uint32_t exc_data_count,uint64_t exc_sub_code,uint64_t exc_sub_sub_code)297 static StopInfoSP GetStopInfoForHardwareBP(Thread &thread, Target *target,
298                                            uint32_t exc_data_count,
299                                            uint64_t exc_sub_code,
300                                            uint64_t exc_sub_sub_code) {
301   // Try hardware watchpoint.
302   if (target) {
303     // The exc_sub_code indicates the data break address.
304     lldb::WatchpointSP wp_sp =
305         target->GetWatchpointList().FindByAddress((lldb::addr_t)exc_sub_code);
306     if (wp_sp && wp_sp->IsEnabled()) {
307       // Debugserver may piggyback the hardware index of the fired watchpoint
308       // in the exception data. Set the hardware index if that's the case.
309       if (exc_data_count >= 3)
310         wp_sp->SetHardwareIndex((uint32_t)exc_sub_sub_code);
311       return StopInfo::CreateStopReasonWithWatchpointID(thread, wp_sp->GetID());
312     }
313   }
314 
315   // Try hardware breakpoint.
316   ProcessSP process_sp(thread.GetProcess());
317   if (process_sp) {
318     // The exc_sub_code indicates the data break address.
319     lldb::BreakpointSiteSP bp_sp =
320         process_sp->GetBreakpointSiteList().FindByAddress(
321             (lldb::addr_t)exc_sub_code);
322     if (bp_sp && bp_sp->IsEnabled()) {
323       // Debugserver may piggyback the hardware index of the fired breakpoint
324       // in the exception data. Set the hardware index if that's the case.
325       if (exc_data_count >= 3)
326         bp_sp->SetHardwareIndex((uint32_t)exc_sub_sub_code);
327       return StopInfo::CreateStopReasonWithBreakpointSiteID(thread,
328                                                             bp_sp->GetID());
329     }
330   }
331 
332   return nullptr;
333 }
334 
CreateStopReasonWithMachException(Thread & thread,uint32_t exc_type,uint32_t exc_data_count,uint64_t exc_code,uint64_t exc_sub_code,uint64_t exc_sub_sub_code,bool pc_already_adjusted,bool adjust_pc_if_needed)335 StopInfoSP StopInfoMachException::CreateStopReasonWithMachException(
336     Thread &thread, uint32_t exc_type, uint32_t exc_data_count,
337     uint64_t exc_code, uint64_t exc_sub_code, uint64_t exc_sub_sub_code,
338     bool pc_already_adjusted, bool adjust_pc_if_needed) {
339   if (exc_type == 0)
340     return StopInfoSP();
341 
342   uint32_t pc_decrement = 0;
343   ExecutionContext exe_ctx(thread.shared_from_this());
344   Target *target = exe_ctx.GetTargetPtr();
345   const llvm::Triple::ArchType cpu =
346       target ? target->GetArchitecture().GetMachine()
347              : llvm::Triple::UnknownArch;
348 
349   switch (exc_type) {
350   case 1: // EXC_BAD_ACCESS
351   case 2: // EXC_BAD_INSTRUCTION
352   case 3: // EXC_ARITHMETIC
353   case 4: // EXC_EMULATION
354     break;
355 
356   case 5:                    // EXC_SOFTWARE
357     if (exc_code == 0x10003) // EXC_SOFT_SIGNAL
358     {
359       if (exc_sub_code == 5) {
360         // On MacOSX, a SIGTRAP can signify that a process has called exec,
361         // so we should check with our dynamic loader to verify.
362         ProcessSP process_sp(thread.GetProcess());
363         if (process_sp) {
364           DynamicLoader *dynamic_loader = process_sp->GetDynamicLoader();
365           if (dynamic_loader && dynamic_loader->ProcessDidExec()) {
366             // The program was re-exec'ed
367             return StopInfo::CreateStopReasonWithExec(thread);
368           }
369         }
370       }
371       return StopInfo::CreateStopReasonWithSignal(thread, exc_sub_code);
372     }
373     break;
374 
375   case 6: // EXC_BREAKPOINT
376   {
377     bool is_actual_breakpoint = false;
378     bool is_trace_if_actual_breakpoint_missing = false;
379     switch (cpu) {
380     case llvm::Triple::x86:
381     case llvm::Triple::x86_64:
382       if (exc_code == 1) // EXC_I386_SGL
383       {
384         if (!exc_sub_code) {
385           // This looks like a plain trap.
386           // Have to check if there is a breakpoint here as well.  When you
387           // single-step onto a trap, the single step stops you not to trap.
388           // Since we also do that check below, let's just use that logic.
389           is_actual_breakpoint = true;
390           is_trace_if_actual_breakpoint_missing = true;
391         } else {
392           if (StopInfoSP stop_info =
393                   GetStopInfoForHardwareBP(thread, target, exc_data_count,
394                                            exc_sub_code, exc_sub_sub_code))
395             return stop_info;
396         }
397       } else if (exc_code == 2 || // EXC_I386_BPT
398                  exc_code == 3)   // EXC_I386_BPTFLT
399       {
400         // KDP returns EXC_I386_BPTFLT for trace breakpoints
401         if (exc_code == 3)
402           is_trace_if_actual_breakpoint_missing = true;
403 
404         is_actual_breakpoint = true;
405         if (!pc_already_adjusted)
406           pc_decrement = 1;
407       }
408       break;
409 
410     case llvm::Triple::arm:
411     case llvm::Triple::thumb:
412       if (exc_code == 0x102) // EXC_ARM_DA_DEBUG
413       {
414         // It's a watchpoint, then, if the exc_sub_code indicates a
415         // known/enabled data break address from our watchpoint list.
416         lldb::WatchpointSP wp_sp;
417         if (target)
418           wp_sp = target->GetWatchpointList().FindByAddress(
419               (lldb::addr_t)exc_sub_code);
420         if (wp_sp && wp_sp->IsEnabled()) {
421           // Debugserver may piggyback the hardware index of the fired
422           // watchpoint in the exception data. Set the hardware index if
423           // that's the case.
424           if (exc_data_count >= 3)
425             wp_sp->SetHardwareIndex((uint32_t)exc_sub_sub_code);
426           return StopInfo::CreateStopReasonWithWatchpointID(thread,
427                                                             wp_sp->GetID());
428         } else {
429           is_actual_breakpoint = true;
430           is_trace_if_actual_breakpoint_missing = true;
431         }
432       } else if (exc_code == 1) // EXC_ARM_BREAKPOINT
433       {
434         is_actual_breakpoint = true;
435         is_trace_if_actual_breakpoint_missing = true;
436       } else if (exc_code == 0) // FIXME not EXC_ARM_BREAKPOINT but a kernel
437                                 // is currently returning this so accept it
438                                 // as indicating a breakpoint until the
439                                 // kernel is fixed
440       {
441         is_actual_breakpoint = true;
442         is_trace_if_actual_breakpoint_missing = true;
443       }
444       break;
445 
446     case llvm::Triple::aarch64_32:
447     case llvm::Triple::aarch64: {
448       if (exc_code == 1 && exc_sub_code == 0) // EXC_ARM_BREAKPOINT
449       {
450         // This is hit when we single instruction step aka MDSCR_EL1 SS bit 0
451         // is set
452         is_actual_breakpoint = false;
453         is_trace_if_actual_breakpoint_missing = true;
454       }
455       if (exc_code == 0x102) // EXC_ARM_DA_DEBUG
456       {
457         // It's a watchpoint, then, if the exc_sub_code indicates a
458         // known/enabled data break address from our watchpoint list.
459         lldb::WatchpointSP wp_sp;
460         if (target)
461           wp_sp = target->GetWatchpointList().FindByAddress(
462               (lldb::addr_t)exc_sub_code);
463         if (wp_sp && wp_sp->IsEnabled()) {
464           // Debugserver may piggyback the hardware index of the fired
465           // watchpoint in the exception data. Set the hardware index if
466           // that's the case.
467           if (exc_data_count >= 3)
468             wp_sp->SetHardwareIndex((uint32_t)exc_sub_sub_code);
469           return StopInfo::CreateStopReasonWithWatchpointID(thread,
470                                                             wp_sp->GetID());
471         }
472         // EXC_ARM_DA_DEBUG seems to be reused for EXC_BREAKPOINT as well as
473         // EXC_BAD_ACCESS
474         if (thread.GetTemporaryResumeState() == eStateStepping)
475           return StopInfo::CreateStopReasonToTrace(thread);
476       }
477       // It looks like exc_sub_code has the 4 bytes of the instruction that
478       // triggered the exception, i.e. our breakpoint opcode
479       is_actual_breakpoint = exc_code == 1;
480       break;
481     }
482 
483     default:
484       break;
485     }
486 
487     if (is_actual_breakpoint) {
488       RegisterContextSP reg_ctx_sp(thread.GetRegisterContext());
489       addr_t pc = reg_ctx_sp->GetPC() - pc_decrement;
490 
491       ProcessSP process_sp(thread.CalculateProcess());
492 
493       lldb::BreakpointSiteSP bp_site_sp;
494       if (process_sp)
495         bp_site_sp = process_sp->GetBreakpointSiteList().FindByAddress(pc);
496       if (bp_site_sp && bp_site_sp->IsEnabled()) {
497         // Update the PC if we were asked to do so, but only do so if we find
498         // a breakpoint that we know about cause this could be a trap
499         // instruction in the code
500         if (pc_decrement > 0 && adjust_pc_if_needed)
501           reg_ctx_sp->SetPC(pc);
502 
503         // If the breakpoint is for this thread, then we'll report the hit,
504         // but if it is for another thread, we can just report no reason.  We
505         // don't need to worry about stepping over the breakpoint here, that
506         // will be taken care of when the thread resumes and notices that
507         // there's a breakpoint under the pc. If we have an operating system
508         // plug-in, we might have set a thread specific breakpoint using the
509         // operating system thread ID, so we can't make any assumptions about
510         // the thread ID so we must always report the breakpoint regardless
511         // of the thread.
512         if (bp_site_sp->ValidForThisThread(&thread) ||
513             thread.GetProcess()->GetOperatingSystem() != nullptr)
514           return StopInfo::CreateStopReasonWithBreakpointSiteID(
515               thread, bp_site_sp->GetID());
516         else if (is_trace_if_actual_breakpoint_missing)
517           return StopInfo::CreateStopReasonToTrace(thread);
518         else
519           return StopInfoSP();
520       }
521 
522       // Don't call this a trace if we weren't single stepping this thread.
523       if (is_trace_if_actual_breakpoint_missing &&
524           thread.GetTemporaryResumeState() == eStateStepping) {
525         return StopInfo::CreateStopReasonToTrace(thread);
526       }
527     }
528   } break;
529 
530   case 7:  // EXC_SYSCALL
531   case 8:  // EXC_MACH_SYSCALL
532   case 9:  // EXC_RPC_ALERT
533   case 10: // EXC_CRASH
534     break;
535   }
536 
537   return StopInfoSP(new StopInfoMachException(thread, exc_type, exc_data_count,
538                                               exc_code, exc_sub_code));
539 }
540