1 //===-- SBBreakpoint.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 "lldb/API/SBBreakpoint.h" 10 #include "lldb/API/SBBreakpointLocation.h" 11 #include "lldb/API/SBDebugger.h" 12 #include "lldb/API/SBEvent.h" 13 #include "lldb/API/SBProcess.h" 14 #include "lldb/API/SBStream.h" 15 #include "lldb/API/SBStringList.h" 16 #include "lldb/API/SBStructuredData.h" 17 #include "lldb/API/SBThread.h" 18 #include "lldb/Utility/Instrumentation.h" 19 20 #include "lldb/Breakpoint/Breakpoint.h" 21 #include "lldb/Breakpoint/BreakpointIDList.h" 22 #include "lldb/Breakpoint/BreakpointLocation.h" 23 #include "lldb/Breakpoint/BreakpointResolver.h" 24 #include "lldb/Breakpoint/BreakpointResolverScripted.h" 25 #include "lldb/Breakpoint/StoppointCallbackContext.h" 26 #include "lldb/Core/Address.h" 27 #include "lldb/Core/Debugger.h" 28 #include "lldb/Core/StructuredDataImpl.h" 29 #include "lldb/Interpreter/CommandInterpreter.h" 30 #include "lldb/Interpreter/ScriptInterpreter.h" 31 #include "lldb/Target/Process.h" 32 #include "lldb/Target/SectionLoadList.h" 33 #include "lldb/Target/Target.h" 34 #include "lldb/Target/Thread.h" 35 #include "lldb/Target/ThreadSpec.h" 36 #include "lldb/Utility/Stream.h" 37 38 #include "SBBreakpointOptionCommon.h" 39 40 #include "lldb/lldb-enumerations.h" 41 42 #include "llvm/ADT/STLExtras.h" 43 44 using namespace lldb; 45 using namespace lldb_private; 46 47 SBBreakpoint::SBBreakpoint() { LLDB_INSTRUMENT_VA(this); } 48 49 SBBreakpoint::SBBreakpoint(const SBBreakpoint &rhs) 50 : m_opaque_wp(rhs.m_opaque_wp) { 51 LLDB_INSTRUMENT_VA(this, rhs); 52 } 53 54 SBBreakpoint::SBBreakpoint(const lldb::BreakpointSP &bp_sp) 55 : m_opaque_wp(bp_sp) { 56 LLDB_INSTRUMENT_VA(this, bp_sp); 57 } 58 59 SBBreakpoint::~SBBreakpoint() = default; 60 61 const SBBreakpoint &SBBreakpoint::operator=(const SBBreakpoint &rhs) { 62 LLDB_INSTRUMENT_VA(this, rhs); 63 64 m_opaque_wp = rhs.m_opaque_wp; 65 return *this; 66 } 67 68 bool SBBreakpoint::operator==(const lldb::SBBreakpoint &rhs) { 69 LLDB_INSTRUMENT_VA(this, rhs); 70 71 return m_opaque_wp.lock() == rhs.m_opaque_wp.lock(); 72 } 73 74 bool SBBreakpoint::operator!=(const lldb::SBBreakpoint &rhs) { 75 LLDB_INSTRUMENT_VA(this, rhs); 76 77 return m_opaque_wp.lock() != rhs.m_opaque_wp.lock(); 78 } 79 80 SBTarget SBBreakpoint::GetTarget() const { 81 LLDB_INSTRUMENT_VA(this); 82 83 BreakpointSP bkpt_sp = GetSP(); 84 if (bkpt_sp) 85 return SBTarget(bkpt_sp->GetTargetSP()); 86 87 return SBTarget(); 88 } 89 90 break_id_t SBBreakpoint::GetID() const { 91 LLDB_INSTRUMENT_VA(this); 92 93 break_id_t break_id = LLDB_INVALID_BREAK_ID; 94 BreakpointSP bkpt_sp = GetSP(); 95 if (bkpt_sp) 96 break_id = bkpt_sp->GetID(); 97 98 return break_id; 99 } 100 101 bool SBBreakpoint::IsValid() const { 102 LLDB_INSTRUMENT_VA(this); 103 return this->operator bool(); 104 } 105 SBBreakpoint::operator bool() const { 106 LLDB_INSTRUMENT_VA(this); 107 108 BreakpointSP bkpt_sp = GetSP(); 109 if (!bkpt_sp) 110 return false; 111 else if (bkpt_sp->GetTarget().GetBreakpointByID(bkpt_sp->GetID())) 112 return true; 113 else 114 return false; 115 } 116 117 void SBBreakpoint::ClearAllBreakpointSites() { 118 LLDB_INSTRUMENT_VA(this); 119 120 BreakpointSP bkpt_sp = GetSP(); 121 if (bkpt_sp) { 122 std::lock_guard<std::recursive_mutex> guard( 123 bkpt_sp->GetTarget().GetAPIMutex()); 124 bkpt_sp->ClearAllBreakpointSites(); 125 } 126 } 127 128 SBBreakpointLocation SBBreakpoint::FindLocationByAddress(addr_t vm_addr) { 129 LLDB_INSTRUMENT_VA(this, vm_addr); 130 131 SBBreakpointLocation sb_bp_location; 132 133 BreakpointSP bkpt_sp = GetSP(); 134 if (bkpt_sp) { 135 if (vm_addr != LLDB_INVALID_ADDRESS) { 136 std::lock_guard<std::recursive_mutex> guard( 137 bkpt_sp->GetTarget().GetAPIMutex()); 138 Address address; 139 Target &target = bkpt_sp->GetTarget(); 140 if (!target.GetSectionLoadList().ResolveLoadAddress(vm_addr, address)) { 141 address.SetRawAddress(vm_addr); 142 } 143 sb_bp_location.SetLocation(bkpt_sp->FindLocationByAddress(address)); 144 } 145 } 146 return sb_bp_location; 147 } 148 149 break_id_t SBBreakpoint::FindLocationIDByAddress(addr_t vm_addr) { 150 LLDB_INSTRUMENT_VA(this, vm_addr); 151 152 break_id_t break_id = LLDB_INVALID_BREAK_ID; 153 BreakpointSP bkpt_sp = GetSP(); 154 155 if (bkpt_sp && vm_addr != LLDB_INVALID_ADDRESS) { 156 std::lock_guard<std::recursive_mutex> guard( 157 bkpt_sp->GetTarget().GetAPIMutex()); 158 Address address; 159 Target &target = bkpt_sp->GetTarget(); 160 if (!target.GetSectionLoadList().ResolveLoadAddress(vm_addr, address)) { 161 address.SetRawAddress(vm_addr); 162 } 163 break_id = bkpt_sp->FindLocationIDByAddress(address); 164 } 165 166 return break_id; 167 } 168 169 SBBreakpointLocation SBBreakpoint::FindLocationByID(break_id_t bp_loc_id) { 170 LLDB_INSTRUMENT_VA(this, bp_loc_id); 171 172 SBBreakpointLocation sb_bp_location; 173 BreakpointSP bkpt_sp = GetSP(); 174 175 if (bkpt_sp) { 176 std::lock_guard<std::recursive_mutex> guard( 177 bkpt_sp->GetTarget().GetAPIMutex()); 178 sb_bp_location.SetLocation(bkpt_sp->FindLocationByID(bp_loc_id)); 179 } 180 181 return sb_bp_location; 182 } 183 184 SBBreakpointLocation SBBreakpoint::GetLocationAtIndex(uint32_t index) { 185 LLDB_INSTRUMENT_VA(this, index); 186 187 SBBreakpointLocation sb_bp_location; 188 BreakpointSP bkpt_sp = GetSP(); 189 190 if (bkpt_sp) { 191 std::lock_guard<std::recursive_mutex> guard( 192 bkpt_sp->GetTarget().GetAPIMutex()); 193 sb_bp_location.SetLocation(bkpt_sp->GetLocationAtIndex(index)); 194 } 195 196 return sb_bp_location; 197 } 198 199 void SBBreakpoint::SetEnabled(bool enable) { 200 LLDB_INSTRUMENT_VA(this, enable); 201 202 BreakpointSP bkpt_sp = GetSP(); 203 204 if (bkpt_sp) { 205 std::lock_guard<std::recursive_mutex> guard( 206 bkpt_sp->GetTarget().GetAPIMutex()); 207 bkpt_sp->SetEnabled(enable); 208 } 209 } 210 211 bool SBBreakpoint::IsEnabled() { 212 LLDB_INSTRUMENT_VA(this); 213 214 BreakpointSP bkpt_sp = GetSP(); 215 if (bkpt_sp) { 216 std::lock_guard<std::recursive_mutex> guard( 217 bkpt_sp->GetTarget().GetAPIMutex()); 218 return bkpt_sp->IsEnabled(); 219 } else 220 return false; 221 } 222 223 void SBBreakpoint::SetOneShot(bool one_shot) { 224 LLDB_INSTRUMENT_VA(this, one_shot); 225 226 BreakpointSP bkpt_sp = GetSP(); 227 228 if (bkpt_sp) { 229 std::lock_guard<std::recursive_mutex> guard( 230 bkpt_sp->GetTarget().GetAPIMutex()); 231 bkpt_sp->SetOneShot(one_shot); 232 } 233 } 234 235 bool SBBreakpoint::IsOneShot() const { 236 LLDB_INSTRUMENT_VA(this); 237 238 BreakpointSP bkpt_sp = GetSP(); 239 if (bkpt_sp) { 240 std::lock_guard<std::recursive_mutex> guard( 241 bkpt_sp->GetTarget().GetAPIMutex()); 242 return bkpt_sp->IsOneShot(); 243 } else 244 return false; 245 } 246 247 bool SBBreakpoint::IsInternal() { 248 LLDB_INSTRUMENT_VA(this); 249 250 BreakpointSP bkpt_sp = GetSP(); 251 if (bkpt_sp) { 252 std::lock_guard<std::recursive_mutex> guard( 253 bkpt_sp->GetTarget().GetAPIMutex()); 254 return bkpt_sp->IsInternal(); 255 } else 256 return false; 257 } 258 259 void SBBreakpoint::SetIgnoreCount(uint32_t count) { 260 LLDB_INSTRUMENT_VA(this, count); 261 262 BreakpointSP bkpt_sp = GetSP(); 263 264 if (bkpt_sp) { 265 std::lock_guard<std::recursive_mutex> guard( 266 bkpt_sp->GetTarget().GetAPIMutex()); 267 bkpt_sp->SetIgnoreCount(count); 268 } 269 } 270 271 void SBBreakpoint::SetCondition(const char *condition) { 272 LLDB_INSTRUMENT_VA(this, condition); 273 274 BreakpointSP bkpt_sp = GetSP(); 275 if (bkpt_sp) { 276 std::lock_guard<std::recursive_mutex> guard( 277 bkpt_sp->GetTarget().GetAPIMutex()); 278 bkpt_sp->SetCondition(condition); 279 } 280 } 281 282 const char *SBBreakpoint::GetCondition() { 283 LLDB_INSTRUMENT_VA(this); 284 285 BreakpointSP bkpt_sp = GetSP(); 286 if (!bkpt_sp) 287 return nullptr; 288 289 std::lock_guard<std::recursive_mutex> guard( 290 bkpt_sp->GetTarget().GetAPIMutex()); 291 return ConstString(bkpt_sp->GetConditionText()).GetCString(); 292 } 293 294 void SBBreakpoint::SetAutoContinue(bool auto_continue) { 295 LLDB_INSTRUMENT_VA(this, auto_continue); 296 297 BreakpointSP bkpt_sp = GetSP(); 298 if (bkpt_sp) { 299 std::lock_guard<std::recursive_mutex> guard( 300 bkpt_sp->GetTarget().GetAPIMutex()); 301 bkpt_sp->SetAutoContinue(auto_continue); 302 } 303 } 304 305 bool SBBreakpoint::GetAutoContinue() { 306 LLDB_INSTRUMENT_VA(this); 307 308 BreakpointSP bkpt_sp = GetSP(); 309 if (bkpt_sp) { 310 std::lock_guard<std::recursive_mutex> guard( 311 bkpt_sp->GetTarget().GetAPIMutex()); 312 return bkpt_sp->IsAutoContinue(); 313 } 314 return false; 315 } 316 317 uint32_t SBBreakpoint::GetHitCount() const { 318 LLDB_INSTRUMENT_VA(this); 319 320 uint32_t count = 0; 321 BreakpointSP bkpt_sp = GetSP(); 322 if (bkpt_sp) { 323 std::lock_guard<std::recursive_mutex> guard( 324 bkpt_sp->GetTarget().GetAPIMutex()); 325 count = bkpt_sp->GetHitCount(); 326 } 327 328 return count; 329 } 330 331 uint32_t SBBreakpoint::GetIgnoreCount() const { 332 LLDB_INSTRUMENT_VA(this); 333 334 uint32_t count = 0; 335 BreakpointSP bkpt_sp = GetSP(); 336 if (bkpt_sp) { 337 std::lock_guard<std::recursive_mutex> guard( 338 bkpt_sp->GetTarget().GetAPIMutex()); 339 count = bkpt_sp->GetIgnoreCount(); 340 } 341 342 return count; 343 } 344 345 void SBBreakpoint::SetThreadID(tid_t tid) { 346 LLDB_INSTRUMENT_VA(this, tid); 347 348 BreakpointSP bkpt_sp = GetSP(); 349 if (bkpt_sp) { 350 std::lock_guard<std::recursive_mutex> guard( 351 bkpt_sp->GetTarget().GetAPIMutex()); 352 bkpt_sp->SetThreadID(tid); 353 } 354 } 355 356 tid_t SBBreakpoint::GetThreadID() { 357 LLDB_INSTRUMENT_VA(this); 358 359 tid_t tid = LLDB_INVALID_THREAD_ID; 360 BreakpointSP bkpt_sp = GetSP(); 361 if (bkpt_sp) { 362 std::lock_guard<std::recursive_mutex> guard( 363 bkpt_sp->GetTarget().GetAPIMutex()); 364 tid = bkpt_sp->GetThreadID(); 365 } 366 367 return tid; 368 } 369 370 void SBBreakpoint::SetThreadIndex(uint32_t index) { 371 LLDB_INSTRUMENT_VA(this, index); 372 373 BreakpointSP bkpt_sp = GetSP(); 374 if (bkpt_sp) { 375 std::lock_guard<std::recursive_mutex> guard( 376 bkpt_sp->GetTarget().GetAPIMutex()); 377 bkpt_sp->GetOptions().GetThreadSpec()->SetIndex(index); 378 } 379 } 380 381 uint32_t SBBreakpoint::GetThreadIndex() const { 382 LLDB_INSTRUMENT_VA(this); 383 384 uint32_t thread_idx = UINT32_MAX; 385 BreakpointSP bkpt_sp = GetSP(); 386 if (bkpt_sp) { 387 std::lock_guard<std::recursive_mutex> guard( 388 bkpt_sp->GetTarget().GetAPIMutex()); 389 const ThreadSpec *thread_spec = 390 bkpt_sp->GetOptions().GetThreadSpecNoCreate(); 391 if (thread_spec != nullptr) 392 thread_idx = thread_spec->GetIndex(); 393 } 394 395 return thread_idx; 396 } 397 398 void SBBreakpoint::SetThreadName(const char *thread_name) { 399 LLDB_INSTRUMENT_VA(this, thread_name); 400 401 BreakpointSP bkpt_sp = GetSP(); 402 403 if (bkpt_sp) { 404 std::lock_guard<std::recursive_mutex> guard( 405 bkpt_sp->GetTarget().GetAPIMutex()); 406 bkpt_sp->GetOptions().GetThreadSpec()->SetName(thread_name); 407 } 408 } 409 410 const char *SBBreakpoint::GetThreadName() const { 411 LLDB_INSTRUMENT_VA(this); 412 413 BreakpointSP bkpt_sp = GetSP(); 414 if (!bkpt_sp) 415 return nullptr; 416 417 std::lock_guard<std::recursive_mutex> guard( 418 bkpt_sp->GetTarget().GetAPIMutex()); 419 if (const ThreadSpec *thread_spec = 420 bkpt_sp->GetOptions().GetThreadSpecNoCreate()) 421 return ConstString(thread_spec->GetName()).GetCString(); 422 423 return nullptr; 424 } 425 426 void SBBreakpoint::SetQueueName(const char *queue_name) { 427 LLDB_INSTRUMENT_VA(this, queue_name); 428 429 BreakpointSP bkpt_sp = GetSP(); 430 if (bkpt_sp) { 431 std::lock_guard<std::recursive_mutex> guard( 432 bkpt_sp->GetTarget().GetAPIMutex()); 433 bkpt_sp->GetOptions().GetThreadSpec()->SetQueueName(queue_name); 434 } 435 } 436 437 const char *SBBreakpoint::GetQueueName() const { 438 LLDB_INSTRUMENT_VA(this); 439 440 BreakpointSP bkpt_sp = GetSP(); 441 if (!bkpt_sp) 442 return nullptr; 443 444 std::lock_guard<std::recursive_mutex> guard( 445 bkpt_sp->GetTarget().GetAPIMutex()); 446 if (const ThreadSpec *thread_spec = 447 bkpt_sp->GetOptions().GetThreadSpecNoCreate()) 448 return ConstString(thread_spec->GetQueueName()).GetCString(); 449 450 return nullptr; 451 } 452 453 size_t SBBreakpoint::GetNumResolvedLocations() const { 454 LLDB_INSTRUMENT_VA(this); 455 456 size_t num_resolved = 0; 457 BreakpointSP bkpt_sp = GetSP(); 458 if (bkpt_sp) { 459 std::lock_guard<std::recursive_mutex> guard( 460 bkpt_sp->GetTarget().GetAPIMutex()); 461 num_resolved = bkpt_sp->GetNumResolvedLocations(); 462 } 463 return num_resolved; 464 } 465 466 size_t SBBreakpoint::GetNumLocations() const { 467 LLDB_INSTRUMENT_VA(this); 468 469 BreakpointSP bkpt_sp = GetSP(); 470 size_t num_locs = 0; 471 if (bkpt_sp) { 472 std::lock_guard<std::recursive_mutex> guard( 473 bkpt_sp->GetTarget().GetAPIMutex()); 474 num_locs = bkpt_sp->GetNumLocations(); 475 } 476 return num_locs; 477 } 478 479 void SBBreakpoint::SetCommandLineCommands(SBStringList &commands) { 480 LLDB_INSTRUMENT_VA(this, commands); 481 482 BreakpointSP bkpt_sp = GetSP(); 483 if (!bkpt_sp) 484 return; 485 if (commands.GetSize() == 0) 486 return; 487 488 std::lock_guard<std::recursive_mutex> guard( 489 bkpt_sp->GetTarget().GetAPIMutex()); 490 std::unique_ptr<BreakpointOptions::CommandData> cmd_data_up( 491 new BreakpointOptions::CommandData(*commands, eScriptLanguageNone)); 492 493 bkpt_sp->GetOptions().SetCommandDataCallback(cmd_data_up); 494 } 495 496 bool SBBreakpoint::GetCommandLineCommands(SBStringList &commands) { 497 LLDB_INSTRUMENT_VA(this, commands); 498 499 BreakpointSP bkpt_sp = GetSP(); 500 if (!bkpt_sp) 501 return false; 502 StringList command_list; 503 bool has_commands = 504 bkpt_sp->GetOptions().GetCommandLineCallbacks(command_list); 505 if (has_commands) 506 commands.AppendList(command_list); 507 return has_commands; 508 } 509 510 bool SBBreakpoint::GetDescription(SBStream &s) { 511 LLDB_INSTRUMENT_VA(this, s); 512 513 return GetDescription(s, true); 514 } 515 516 bool SBBreakpoint::GetDescription(SBStream &s, bool include_locations) { 517 LLDB_INSTRUMENT_VA(this, s, include_locations); 518 519 BreakpointSP bkpt_sp = GetSP(); 520 if (bkpt_sp) { 521 std::lock_guard<std::recursive_mutex> guard( 522 bkpt_sp->GetTarget().GetAPIMutex()); 523 s.Printf("SBBreakpoint: id = %i, ", bkpt_sp->GetID()); 524 bkpt_sp->GetResolverDescription(s.get()); 525 bkpt_sp->GetFilterDescription(s.get()); 526 if (include_locations) { 527 const size_t num_locations = bkpt_sp->GetNumLocations(); 528 s.Printf(", locations = %" PRIu64, (uint64_t)num_locations); 529 } 530 return true; 531 } 532 s.Printf("No value"); 533 return false; 534 } 535 536 SBError SBBreakpoint::AddLocation(SBAddress &address) { 537 LLDB_INSTRUMENT_VA(this, address); 538 539 BreakpointSP bkpt_sp = GetSP(); 540 SBError error; 541 542 if (!address.IsValid()) { 543 error.SetErrorString("Can't add an invalid address."); 544 return error; 545 } 546 547 if (!bkpt_sp) { 548 error.SetErrorString("No breakpoint to add a location to."); 549 return error; 550 } 551 552 if (!llvm::isa<BreakpointResolverScripted>(bkpt_sp->GetResolver().get())) { 553 error.SetErrorString("Only a scripted resolver can add locations."); 554 return error; 555 } 556 557 if (bkpt_sp->GetSearchFilter()->AddressPasses(address.ref())) 558 bkpt_sp->AddLocation(address.ref()); 559 else { 560 StreamString s; 561 address.get()->Dump(&s, &bkpt_sp->GetTarget(), 562 Address::DumpStyleModuleWithFileAddress); 563 error.SetErrorStringWithFormat("Address: %s didn't pass the filter.", 564 s.GetData()); 565 } 566 return error; 567 } 568 569 SBStructuredData SBBreakpoint::SerializeToStructuredData() { 570 LLDB_INSTRUMENT_VA(this); 571 572 SBStructuredData data; 573 BreakpointSP bkpt_sp = GetSP(); 574 575 if (!bkpt_sp) 576 return data; 577 578 StructuredData::ObjectSP bkpt_dict = bkpt_sp->SerializeToStructuredData(); 579 data.m_impl_up->SetObjectSP(bkpt_dict); 580 return data; 581 } 582 583 void SBBreakpoint::SetCallback(SBBreakpointHitCallback callback, void *baton) { 584 LLDB_INSTRUMENT_VA(this, callback, baton); 585 586 BreakpointSP bkpt_sp = GetSP(); 587 588 if (bkpt_sp) { 589 std::lock_guard<std::recursive_mutex> guard( 590 bkpt_sp->GetTarget().GetAPIMutex()); 591 BatonSP baton_sp(new SBBreakpointCallbackBaton(callback, baton)); 592 bkpt_sp->SetCallback(SBBreakpointCallbackBaton 593 ::PrivateBreakpointHitCallback, baton_sp, 594 false); 595 } 596 } 597 598 void SBBreakpoint::SetScriptCallbackFunction( 599 const char *callback_function_name) { 600 LLDB_INSTRUMENT_VA(this, callback_function_name); 601 SBStructuredData empty_args; 602 SetScriptCallbackFunction(callback_function_name, empty_args); 603 } 604 605 SBError SBBreakpoint::SetScriptCallbackFunction( 606 const char *callback_function_name, 607 SBStructuredData &extra_args) { 608 LLDB_INSTRUMENT_VA(this, callback_function_name, extra_args); 609 SBError sb_error; 610 BreakpointSP bkpt_sp = GetSP(); 611 612 if (bkpt_sp) { 613 Status error; 614 std::lock_guard<std::recursive_mutex> guard( 615 bkpt_sp->GetTarget().GetAPIMutex()); 616 BreakpointOptions &bp_options = bkpt_sp->GetOptions(); 617 error = bkpt_sp->GetTarget() 618 .GetDebugger() 619 .GetScriptInterpreter() 620 ->SetBreakpointCommandCallbackFunction(bp_options, 621 callback_function_name, 622 extra_args.m_impl_up 623 ->GetObjectSP()); 624 sb_error.SetError(error); 625 } else 626 sb_error.SetErrorString("invalid breakpoint"); 627 628 return sb_error; 629 } 630 631 SBError SBBreakpoint::SetScriptCallbackBody(const char *callback_body_text) { 632 LLDB_INSTRUMENT_VA(this, callback_body_text); 633 634 BreakpointSP bkpt_sp = GetSP(); 635 636 SBError sb_error; 637 if (bkpt_sp) { 638 std::lock_guard<std::recursive_mutex> guard( 639 bkpt_sp->GetTarget().GetAPIMutex()); 640 BreakpointOptions &bp_options = bkpt_sp->GetOptions(); 641 Status error = 642 bkpt_sp->GetTarget() 643 .GetDebugger() 644 .GetScriptInterpreter() 645 ->SetBreakpointCommandCallback(bp_options, callback_body_text, 646 /*is_callback=*/false); 647 sb_error.SetError(error); 648 } else 649 sb_error.SetErrorString("invalid breakpoint"); 650 651 return sb_error; 652 } 653 654 bool SBBreakpoint::AddName(const char *new_name) { 655 LLDB_INSTRUMENT_VA(this, new_name); 656 657 SBError status = AddNameWithErrorHandling(new_name); 658 return status.Success(); 659 } 660 661 SBError SBBreakpoint::AddNameWithErrorHandling(const char *new_name) { 662 LLDB_INSTRUMENT_VA(this, new_name); 663 664 BreakpointSP bkpt_sp = GetSP(); 665 666 SBError status; 667 if (bkpt_sp) { 668 std::lock_guard<std::recursive_mutex> guard( 669 bkpt_sp->GetTarget().GetAPIMutex()); 670 Status error; 671 bkpt_sp->GetTarget().AddNameToBreakpoint(bkpt_sp, new_name, error); 672 status.SetError(error); 673 } else { 674 status.SetErrorString("invalid breakpoint"); 675 } 676 677 return status; 678 } 679 680 void SBBreakpoint::RemoveName(const char *name_to_remove) { 681 LLDB_INSTRUMENT_VA(this, name_to_remove); 682 683 BreakpointSP bkpt_sp = GetSP(); 684 685 if (bkpt_sp) { 686 std::lock_guard<std::recursive_mutex> guard( 687 bkpt_sp->GetTarget().GetAPIMutex()); 688 bkpt_sp->GetTarget().RemoveNameFromBreakpoint(bkpt_sp, 689 ConstString(name_to_remove)); 690 } 691 } 692 693 bool SBBreakpoint::MatchesName(const char *name) { 694 LLDB_INSTRUMENT_VA(this, name); 695 696 BreakpointSP bkpt_sp = GetSP(); 697 698 if (bkpt_sp) { 699 std::lock_guard<std::recursive_mutex> guard( 700 bkpt_sp->GetTarget().GetAPIMutex()); 701 return bkpt_sp->MatchesName(name); 702 } 703 704 return false; 705 } 706 707 void SBBreakpoint::GetNames(SBStringList &names) { 708 LLDB_INSTRUMENT_VA(this, names); 709 710 BreakpointSP bkpt_sp = GetSP(); 711 712 if (bkpt_sp) { 713 std::lock_guard<std::recursive_mutex> guard( 714 bkpt_sp->GetTarget().GetAPIMutex()); 715 std::vector<std::string> names_vec; 716 bkpt_sp->GetNames(names_vec); 717 for (const std::string &name : names_vec) { 718 names.AppendString(name.c_str()); 719 } 720 } 721 } 722 723 bool SBBreakpoint::EventIsBreakpointEvent(const lldb::SBEvent &event) { 724 LLDB_INSTRUMENT_VA(event); 725 726 return Breakpoint::BreakpointEventData::GetEventDataFromEvent(event.get()) != 727 nullptr; 728 } 729 730 BreakpointEventType 731 SBBreakpoint::GetBreakpointEventTypeFromEvent(const SBEvent &event) { 732 LLDB_INSTRUMENT_VA(event); 733 734 if (event.IsValid()) 735 return Breakpoint::BreakpointEventData::GetBreakpointEventTypeFromEvent( 736 event.GetSP()); 737 return eBreakpointEventTypeInvalidType; 738 } 739 740 SBBreakpoint SBBreakpoint::GetBreakpointFromEvent(const lldb::SBEvent &event) { 741 LLDB_INSTRUMENT_VA(event); 742 743 if (event.IsValid()) 744 return SBBreakpoint( 745 Breakpoint::BreakpointEventData::GetBreakpointFromEvent(event.GetSP())); 746 return SBBreakpoint(); 747 } 748 749 SBBreakpointLocation 750 SBBreakpoint::GetBreakpointLocationAtIndexFromEvent(const lldb::SBEvent &event, 751 uint32_t loc_idx) { 752 LLDB_INSTRUMENT_VA(event, loc_idx); 753 754 SBBreakpointLocation sb_breakpoint_loc; 755 if (event.IsValid()) 756 sb_breakpoint_loc.SetLocation( 757 Breakpoint::BreakpointEventData::GetBreakpointLocationAtIndexFromEvent( 758 event.GetSP(), loc_idx)); 759 return sb_breakpoint_loc; 760 } 761 762 uint32_t 763 SBBreakpoint::GetNumBreakpointLocationsFromEvent(const lldb::SBEvent &event) { 764 LLDB_INSTRUMENT_VA(event); 765 766 uint32_t num_locations = 0; 767 if (event.IsValid()) 768 num_locations = 769 (Breakpoint::BreakpointEventData::GetNumBreakpointLocationsFromEvent( 770 event.GetSP())); 771 return num_locations; 772 } 773 774 bool SBBreakpoint::IsHardware() const { 775 LLDB_INSTRUMENT_VA(this); 776 777 BreakpointSP bkpt_sp = GetSP(); 778 if (bkpt_sp) 779 return bkpt_sp->IsHardware(); 780 return false; 781 } 782 783 BreakpointSP SBBreakpoint::GetSP() const { return m_opaque_wp.lock(); } 784 785 // This is simple collection of breakpoint id's and their target. 786 class SBBreakpointListImpl { 787 public: 788 SBBreakpointListImpl(lldb::TargetSP target_sp) { 789 if (target_sp && target_sp->IsValid()) 790 m_target_wp = target_sp; 791 } 792 793 ~SBBreakpointListImpl() = default; 794 795 size_t GetSize() { return m_break_ids.size(); } 796 797 BreakpointSP GetBreakpointAtIndex(size_t idx) { 798 if (idx >= m_break_ids.size()) 799 return BreakpointSP(); 800 TargetSP target_sp = m_target_wp.lock(); 801 if (!target_sp) 802 return BreakpointSP(); 803 lldb::break_id_t bp_id = m_break_ids[idx]; 804 return target_sp->GetBreakpointList().FindBreakpointByID(bp_id); 805 } 806 807 BreakpointSP FindBreakpointByID(lldb::break_id_t desired_id) { 808 TargetSP target_sp = m_target_wp.lock(); 809 if (!target_sp) 810 return BreakpointSP(); 811 812 for (lldb::break_id_t &break_id : m_break_ids) { 813 if (break_id == desired_id) 814 return target_sp->GetBreakpointList().FindBreakpointByID(break_id); 815 } 816 return BreakpointSP(); 817 } 818 819 bool Append(BreakpointSP bkpt) { 820 TargetSP target_sp = m_target_wp.lock(); 821 if (!target_sp || !bkpt) 822 return false; 823 if (bkpt->GetTargetSP() != target_sp) 824 return false; 825 m_break_ids.push_back(bkpt->GetID()); 826 return true; 827 } 828 829 bool AppendIfUnique(BreakpointSP bkpt) { 830 TargetSP target_sp = m_target_wp.lock(); 831 if (!target_sp || !bkpt) 832 return false; 833 if (bkpt->GetTargetSP() != target_sp) 834 return false; 835 lldb::break_id_t bp_id = bkpt->GetID(); 836 if (!llvm::is_contained(m_break_ids, bp_id)) 837 return false; 838 839 m_break_ids.push_back(bkpt->GetID()); 840 return true; 841 } 842 843 bool AppendByID(lldb::break_id_t id) { 844 TargetSP target_sp = m_target_wp.lock(); 845 if (!target_sp) 846 return false; 847 if (id == LLDB_INVALID_BREAK_ID) 848 return false; 849 m_break_ids.push_back(id); 850 return true; 851 } 852 853 void Clear() { m_break_ids.clear(); } 854 855 void CopyToBreakpointIDList(lldb_private::BreakpointIDList &bp_list) { 856 for (lldb::break_id_t id : m_break_ids) { 857 bp_list.AddBreakpointID(BreakpointID(id)); 858 } 859 } 860 861 TargetSP GetTarget() { return m_target_wp.lock(); } 862 863 private: 864 std::vector<lldb::break_id_t> m_break_ids; 865 TargetWP m_target_wp; 866 }; 867 868 SBBreakpointList::SBBreakpointList(SBTarget &target) 869 : m_opaque_sp(new SBBreakpointListImpl(target.GetSP())) { 870 LLDB_INSTRUMENT_VA(this, target); 871 } 872 873 SBBreakpointList::~SBBreakpointList() = default; 874 875 size_t SBBreakpointList::GetSize() const { 876 LLDB_INSTRUMENT_VA(this); 877 878 if (!m_opaque_sp) 879 return 0; 880 else 881 return m_opaque_sp->GetSize(); 882 } 883 884 SBBreakpoint SBBreakpointList::GetBreakpointAtIndex(size_t idx) { 885 LLDB_INSTRUMENT_VA(this, idx); 886 887 if (!m_opaque_sp) 888 return SBBreakpoint(); 889 890 BreakpointSP bkpt_sp = m_opaque_sp->GetBreakpointAtIndex(idx); 891 return SBBreakpoint(bkpt_sp); 892 } 893 894 SBBreakpoint SBBreakpointList::FindBreakpointByID(lldb::break_id_t id) { 895 LLDB_INSTRUMENT_VA(this, id); 896 897 if (!m_opaque_sp) 898 return SBBreakpoint(); 899 BreakpointSP bkpt_sp = m_opaque_sp->FindBreakpointByID(id); 900 return SBBreakpoint(bkpt_sp); 901 } 902 903 void SBBreakpointList::Append(const SBBreakpoint &sb_bkpt) { 904 LLDB_INSTRUMENT_VA(this, sb_bkpt); 905 906 if (!sb_bkpt.IsValid()) 907 return; 908 if (!m_opaque_sp) 909 return; 910 m_opaque_sp->Append(sb_bkpt.m_opaque_wp.lock()); 911 } 912 913 void SBBreakpointList::AppendByID(lldb::break_id_t id) { 914 LLDB_INSTRUMENT_VA(this, id); 915 916 if (!m_opaque_sp) 917 return; 918 m_opaque_sp->AppendByID(id); 919 } 920 921 bool SBBreakpointList::AppendIfUnique(const SBBreakpoint &sb_bkpt) { 922 LLDB_INSTRUMENT_VA(this, sb_bkpt); 923 924 if (!sb_bkpt.IsValid()) 925 return false; 926 if (!m_opaque_sp) 927 return false; 928 return m_opaque_sp->AppendIfUnique(sb_bkpt.GetSP()); 929 } 930 931 void SBBreakpointList::Clear() { 932 LLDB_INSTRUMENT_VA(this); 933 934 if (m_opaque_sp) 935 m_opaque_sp->Clear(); 936 } 937 938 void SBBreakpointList::CopyToBreakpointIDList( 939 lldb_private::BreakpointIDList &bp_id_list) { 940 if (m_opaque_sp) 941 m_opaque_sp->CopyToBreakpointIDList(bp_id_list); 942 } 943