1 //===-- ABISysV_arm.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 "ABISysV_arm.h" 10 11 #include <vector> 12 13 #include "llvm/ADT/STLExtras.h" 14 #include "llvm/ADT/Triple.h" 15 16 #include "lldb/Core/Module.h" 17 #include "lldb/Core/PluginManager.h" 18 #include "lldb/Core/Value.h" 19 #include "lldb/Core/ValueObjectConstResult.h" 20 #include "lldb/Symbol/UnwindPlan.h" 21 #include "lldb/Target/Process.h" 22 #include "lldb/Target/RegisterContext.h" 23 #include "lldb/Target/Target.h" 24 #include "lldb/Target/Thread.h" 25 #include "lldb/Utility/ConstString.h" 26 #include "lldb/Utility/RegisterValue.h" 27 #include "lldb/Utility/Scalar.h" 28 #include "lldb/Utility/Status.h" 29 30 #include "Plugins/Process/Utility/ARMDefines.h" 31 #include "Utility/ARM_DWARF_Registers.h" 32 #include "Utility/ARM_ehframe_Registers.h" 33 34 using namespace lldb; 35 using namespace lldb_private; 36 37 LLDB_PLUGIN_DEFINE(ABISysV_arm) 38 39 static const RegisterInfo g_register_infos[] = { 40 // NAME ALT SZ OFF ENCODING FORMAT EH_FRAME 41 // DWARF GENERIC PROCESS PLUGIN 42 // LLDB NATIVE VALUE REGS INVALIDATE REGS 43 // ========== ======= == === ============= ============ 44 // ======================= =================== =========================== 45 // ======================= ====================== ========== 46 // =============== 47 {"r0", 48 "arg1", 49 4, 50 0, 51 eEncodingUint, 52 eFormatHex, 53 {ehframe_r0, dwarf_r0, LLDB_REGNUM_GENERIC_ARG1, LLDB_INVALID_REGNUM, 54 LLDB_INVALID_REGNUM}, 55 nullptr, 56 nullptr, 57 nullptr, 58 0}, 59 {"r1", 60 "arg2", 61 4, 62 0, 63 eEncodingUint, 64 eFormatHex, 65 {ehframe_r1, dwarf_r1, LLDB_REGNUM_GENERIC_ARG2, LLDB_INVALID_REGNUM, 66 LLDB_INVALID_REGNUM}, 67 nullptr, 68 nullptr, 69 nullptr, 70 0}, 71 {"r2", 72 "arg3", 73 4, 74 0, 75 eEncodingUint, 76 eFormatHex, 77 {ehframe_r2, dwarf_r2, LLDB_REGNUM_GENERIC_ARG3, LLDB_INVALID_REGNUM, 78 LLDB_INVALID_REGNUM}, 79 nullptr, 80 nullptr, 81 nullptr, 82 0}, 83 {"r3", 84 "arg4", 85 4, 86 0, 87 eEncodingUint, 88 eFormatHex, 89 {ehframe_r3, dwarf_r3, LLDB_REGNUM_GENERIC_ARG4, LLDB_INVALID_REGNUM, 90 LLDB_INVALID_REGNUM}, 91 nullptr, 92 nullptr, 93 nullptr, 94 0}, 95 {"r4", 96 nullptr, 97 4, 98 0, 99 eEncodingUint, 100 eFormatHex, 101 {ehframe_r4, dwarf_r4, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 102 LLDB_INVALID_REGNUM}, 103 nullptr, 104 nullptr, 105 nullptr, 106 0}, 107 {"r5", 108 nullptr, 109 4, 110 0, 111 eEncodingUint, 112 eFormatHex, 113 {ehframe_r5, dwarf_r5, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 114 LLDB_INVALID_REGNUM}, 115 nullptr, 116 nullptr, 117 nullptr, 118 0}, 119 {"r6", 120 nullptr, 121 4, 122 0, 123 eEncodingUint, 124 eFormatHex, 125 {ehframe_r6, dwarf_r6, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 126 LLDB_INVALID_REGNUM}, 127 nullptr, 128 nullptr, 129 nullptr, 130 0}, 131 {"r7", 132 nullptr, 133 4, 134 0, 135 eEncodingUint, 136 eFormatHex, 137 {ehframe_r7, dwarf_r7, LLDB_REGNUM_GENERIC_FP, LLDB_INVALID_REGNUM, 138 LLDB_INVALID_REGNUM}, 139 nullptr, 140 nullptr, 141 nullptr, 142 0}, 143 {"r8", 144 nullptr, 145 4, 146 0, 147 eEncodingUint, 148 eFormatHex, 149 {ehframe_r8, dwarf_r8, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 150 LLDB_INVALID_REGNUM}, 151 nullptr, 152 nullptr, 153 nullptr, 154 0}, 155 {"r9", 156 nullptr, 157 4, 158 0, 159 eEncodingUint, 160 eFormatHex, 161 {ehframe_r9, dwarf_r9, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 162 LLDB_INVALID_REGNUM}, 163 nullptr, 164 nullptr, 165 nullptr, 166 0}, 167 {"r10", 168 nullptr, 169 4, 170 0, 171 eEncodingUint, 172 eFormatHex, 173 {ehframe_r10, dwarf_r10, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 174 LLDB_INVALID_REGNUM}, 175 nullptr, 176 nullptr, 177 nullptr, 178 0}, 179 {"r11", 180 nullptr, 181 4, 182 0, 183 eEncodingUint, 184 eFormatHex, 185 {ehframe_r11, dwarf_r11, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 186 LLDB_INVALID_REGNUM}, 187 nullptr, 188 nullptr, 189 nullptr, 190 0}, 191 {"r12", 192 nullptr, 193 4, 194 0, 195 eEncodingUint, 196 eFormatHex, 197 {ehframe_r12, dwarf_r12, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 198 LLDB_INVALID_REGNUM}, 199 nullptr, 200 nullptr, 201 nullptr, 202 0}, 203 {"sp", 204 "r13", 205 4, 206 0, 207 eEncodingUint, 208 eFormatHex, 209 {ehframe_sp, dwarf_sp, LLDB_REGNUM_GENERIC_SP, LLDB_INVALID_REGNUM, 210 LLDB_INVALID_REGNUM}, 211 nullptr, 212 nullptr, 213 nullptr, 214 0}, 215 {"lr", 216 "r14", 217 4, 218 0, 219 eEncodingUint, 220 eFormatHex, 221 {ehframe_lr, dwarf_lr, LLDB_REGNUM_GENERIC_RA, LLDB_INVALID_REGNUM, 222 LLDB_INVALID_REGNUM}, 223 nullptr, 224 nullptr, 225 nullptr, 226 0}, 227 {"pc", 228 "r15", 229 4, 230 0, 231 eEncodingUint, 232 eFormatHex, 233 {ehframe_pc, dwarf_pc, LLDB_REGNUM_GENERIC_PC, LLDB_INVALID_REGNUM, 234 LLDB_INVALID_REGNUM}, 235 nullptr, 236 nullptr, 237 nullptr, 238 0}, 239 {"cpsr", 240 "psr", 241 4, 242 0, 243 eEncodingUint, 244 eFormatHex, 245 {ehframe_cpsr, dwarf_cpsr, LLDB_REGNUM_GENERIC_FLAGS, LLDB_INVALID_REGNUM, 246 LLDB_INVALID_REGNUM}, 247 nullptr, 248 nullptr, 249 nullptr, 250 0}, 251 {"s0", 252 nullptr, 253 4, 254 0, 255 eEncodingIEEE754, 256 eFormatFloat, 257 {LLDB_INVALID_REGNUM, dwarf_s0, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 258 LLDB_INVALID_REGNUM}, 259 nullptr, 260 nullptr, 261 nullptr, 262 0}, 263 {"s1", 264 nullptr, 265 4, 266 0, 267 eEncodingIEEE754, 268 eFormatFloat, 269 {LLDB_INVALID_REGNUM, dwarf_s1, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 270 LLDB_INVALID_REGNUM}, 271 nullptr, 272 nullptr, 273 nullptr, 274 0}, 275 {"s2", 276 nullptr, 277 4, 278 0, 279 eEncodingIEEE754, 280 eFormatFloat, 281 {LLDB_INVALID_REGNUM, dwarf_s2, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 282 LLDB_INVALID_REGNUM}, 283 nullptr, 284 nullptr, 285 nullptr, 286 0}, 287 {"s3", 288 nullptr, 289 4, 290 0, 291 eEncodingIEEE754, 292 eFormatFloat, 293 {LLDB_INVALID_REGNUM, dwarf_s3, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 294 LLDB_INVALID_REGNUM}, 295 nullptr, 296 nullptr, 297 nullptr, 298 0}, 299 {"s4", 300 nullptr, 301 4, 302 0, 303 eEncodingIEEE754, 304 eFormatFloat, 305 {LLDB_INVALID_REGNUM, dwarf_s4, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 306 LLDB_INVALID_REGNUM}, 307 nullptr, 308 nullptr, 309 nullptr, 310 0}, 311 {"s5", 312 nullptr, 313 4, 314 0, 315 eEncodingIEEE754, 316 eFormatFloat, 317 {LLDB_INVALID_REGNUM, dwarf_s5, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 318 LLDB_INVALID_REGNUM}, 319 nullptr, 320 nullptr, 321 nullptr, 322 0}, 323 {"s6", 324 nullptr, 325 4, 326 0, 327 eEncodingIEEE754, 328 eFormatFloat, 329 {LLDB_INVALID_REGNUM, dwarf_s6, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 330 LLDB_INVALID_REGNUM}, 331 nullptr, 332 nullptr, 333 nullptr, 334 0}, 335 {"s7", 336 nullptr, 337 4, 338 0, 339 eEncodingIEEE754, 340 eFormatFloat, 341 {LLDB_INVALID_REGNUM, dwarf_s7, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 342 LLDB_INVALID_REGNUM}, 343 nullptr, 344 nullptr, 345 nullptr, 346 0}, 347 {"s8", 348 nullptr, 349 4, 350 0, 351 eEncodingIEEE754, 352 eFormatFloat, 353 {LLDB_INVALID_REGNUM, dwarf_s8, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 354 LLDB_INVALID_REGNUM}, 355 nullptr, 356 nullptr, 357 nullptr, 358 0}, 359 {"s9", 360 nullptr, 361 4, 362 0, 363 eEncodingIEEE754, 364 eFormatFloat, 365 {LLDB_INVALID_REGNUM, dwarf_s9, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 366 LLDB_INVALID_REGNUM}, 367 nullptr, 368 nullptr, 369 nullptr, 370 0}, 371 {"s10", 372 nullptr, 373 4, 374 0, 375 eEncodingIEEE754, 376 eFormatFloat, 377 {LLDB_INVALID_REGNUM, dwarf_s10, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 378 LLDB_INVALID_REGNUM}, 379 nullptr, 380 nullptr, 381 nullptr, 382 0}, 383 {"s11", 384 nullptr, 385 4, 386 0, 387 eEncodingIEEE754, 388 eFormatFloat, 389 {LLDB_INVALID_REGNUM, dwarf_s11, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 390 LLDB_INVALID_REGNUM}, 391 nullptr, 392 nullptr, 393 nullptr, 394 0}, 395 {"s12", 396 nullptr, 397 4, 398 0, 399 eEncodingIEEE754, 400 eFormatFloat, 401 {LLDB_INVALID_REGNUM, dwarf_s12, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 402 LLDB_INVALID_REGNUM}, 403 nullptr, 404 nullptr, 405 nullptr, 406 0}, 407 {"s13", 408 nullptr, 409 4, 410 0, 411 eEncodingIEEE754, 412 eFormatFloat, 413 {LLDB_INVALID_REGNUM, dwarf_s13, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 414 LLDB_INVALID_REGNUM}, 415 nullptr, 416 nullptr, 417 nullptr, 418 0}, 419 {"s14", 420 nullptr, 421 4, 422 0, 423 eEncodingIEEE754, 424 eFormatFloat, 425 {LLDB_INVALID_REGNUM, dwarf_s14, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 426 LLDB_INVALID_REGNUM}, 427 nullptr, 428 nullptr, 429 nullptr, 430 0}, 431 {"s15", 432 nullptr, 433 4, 434 0, 435 eEncodingIEEE754, 436 eFormatFloat, 437 {LLDB_INVALID_REGNUM, dwarf_s15, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 438 LLDB_INVALID_REGNUM}, 439 nullptr, 440 nullptr, 441 nullptr, 442 0}, 443 {"s16", 444 nullptr, 445 4, 446 0, 447 eEncodingIEEE754, 448 eFormatFloat, 449 {LLDB_INVALID_REGNUM, dwarf_s16, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 450 LLDB_INVALID_REGNUM}, 451 nullptr, 452 nullptr, 453 nullptr, 454 0}, 455 {"s17", 456 nullptr, 457 4, 458 0, 459 eEncodingIEEE754, 460 eFormatFloat, 461 {LLDB_INVALID_REGNUM, dwarf_s17, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 462 LLDB_INVALID_REGNUM}, 463 nullptr, 464 nullptr, 465 nullptr, 466 0}, 467 {"s18", 468 nullptr, 469 4, 470 0, 471 eEncodingIEEE754, 472 eFormatFloat, 473 {LLDB_INVALID_REGNUM, dwarf_s18, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 474 LLDB_INVALID_REGNUM}, 475 nullptr, 476 nullptr, 477 nullptr, 478 0}, 479 {"s19", 480 nullptr, 481 4, 482 0, 483 eEncodingIEEE754, 484 eFormatFloat, 485 {LLDB_INVALID_REGNUM, dwarf_s19, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 486 LLDB_INVALID_REGNUM}, 487 nullptr, 488 nullptr, 489 nullptr, 490 0}, 491 {"s20", 492 nullptr, 493 4, 494 0, 495 eEncodingIEEE754, 496 eFormatFloat, 497 {LLDB_INVALID_REGNUM, dwarf_s20, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 498 LLDB_INVALID_REGNUM}, 499 nullptr, 500 nullptr, 501 nullptr, 502 0}, 503 {"s21", 504 nullptr, 505 4, 506 0, 507 eEncodingIEEE754, 508 eFormatFloat, 509 {LLDB_INVALID_REGNUM, dwarf_s21, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 510 LLDB_INVALID_REGNUM}, 511 nullptr, 512 nullptr, 513 nullptr, 514 0}, 515 {"s22", 516 nullptr, 517 4, 518 0, 519 eEncodingIEEE754, 520 eFormatFloat, 521 {LLDB_INVALID_REGNUM, dwarf_s22, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 522 LLDB_INVALID_REGNUM}, 523 nullptr, 524 nullptr, 525 nullptr, 526 0}, 527 {"s23", 528 nullptr, 529 4, 530 0, 531 eEncodingIEEE754, 532 eFormatFloat, 533 {LLDB_INVALID_REGNUM, dwarf_s23, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 534 LLDB_INVALID_REGNUM}, 535 nullptr, 536 nullptr, 537 nullptr, 538 0}, 539 {"s24", 540 nullptr, 541 4, 542 0, 543 eEncodingIEEE754, 544 eFormatFloat, 545 {LLDB_INVALID_REGNUM, dwarf_s24, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 546 LLDB_INVALID_REGNUM}, 547 nullptr, 548 nullptr, 549 nullptr, 550 0}, 551 {"s25", 552 nullptr, 553 4, 554 0, 555 eEncodingIEEE754, 556 eFormatFloat, 557 {LLDB_INVALID_REGNUM, dwarf_s25, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 558 LLDB_INVALID_REGNUM}, 559 nullptr, 560 nullptr, 561 nullptr, 562 0}, 563 {"s26", 564 nullptr, 565 4, 566 0, 567 eEncodingIEEE754, 568 eFormatFloat, 569 {LLDB_INVALID_REGNUM, dwarf_s26, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 570 LLDB_INVALID_REGNUM}, 571 nullptr, 572 nullptr, 573 nullptr, 574 0}, 575 {"s27", 576 nullptr, 577 4, 578 0, 579 eEncodingIEEE754, 580 eFormatFloat, 581 {LLDB_INVALID_REGNUM, dwarf_s27, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 582 LLDB_INVALID_REGNUM}, 583 nullptr, 584 nullptr, 585 nullptr, 586 0}, 587 {"s28", 588 nullptr, 589 4, 590 0, 591 eEncodingIEEE754, 592 eFormatFloat, 593 {LLDB_INVALID_REGNUM, dwarf_s28, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 594 LLDB_INVALID_REGNUM}, 595 nullptr, 596 nullptr, 597 nullptr, 598 0}, 599 {"s29", 600 nullptr, 601 4, 602 0, 603 eEncodingIEEE754, 604 eFormatFloat, 605 {LLDB_INVALID_REGNUM, dwarf_s29, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 606 LLDB_INVALID_REGNUM}, 607 nullptr, 608 nullptr, 609 nullptr, 610 0}, 611 {"s30", 612 nullptr, 613 4, 614 0, 615 eEncodingIEEE754, 616 eFormatFloat, 617 {LLDB_INVALID_REGNUM, dwarf_s30, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 618 LLDB_INVALID_REGNUM}, 619 nullptr, 620 nullptr, 621 nullptr, 622 0}, 623 {"s31", 624 nullptr, 625 4, 626 0, 627 eEncodingIEEE754, 628 eFormatFloat, 629 {LLDB_INVALID_REGNUM, dwarf_s31, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 630 LLDB_INVALID_REGNUM}, 631 nullptr, 632 nullptr, 633 nullptr, 634 0}, 635 {"fpscr", 636 nullptr, 637 4, 638 0, 639 eEncodingUint, 640 eFormatHex, 641 {LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 642 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 643 nullptr, 644 nullptr, 645 nullptr, 646 0}, 647 {"d0", 648 nullptr, 649 8, 650 0, 651 eEncodingIEEE754, 652 eFormatFloat, 653 {LLDB_INVALID_REGNUM, dwarf_d0, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 654 LLDB_INVALID_REGNUM}, 655 nullptr, 656 nullptr, 657 nullptr, 658 0}, 659 {"d1", 660 nullptr, 661 8, 662 0, 663 eEncodingIEEE754, 664 eFormatFloat, 665 {LLDB_INVALID_REGNUM, dwarf_d1, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 666 LLDB_INVALID_REGNUM}, 667 nullptr, 668 nullptr, 669 nullptr, 670 0}, 671 {"d2", 672 nullptr, 673 8, 674 0, 675 eEncodingIEEE754, 676 eFormatFloat, 677 {LLDB_INVALID_REGNUM, dwarf_d2, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 678 LLDB_INVALID_REGNUM}, 679 nullptr, 680 nullptr, 681 nullptr, 682 0}, 683 {"d3", 684 nullptr, 685 8, 686 0, 687 eEncodingIEEE754, 688 eFormatFloat, 689 {LLDB_INVALID_REGNUM, dwarf_d3, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 690 LLDB_INVALID_REGNUM}, 691 nullptr, 692 nullptr, 693 nullptr, 694 0}, 695 {"d4", 696 nullptr, 697 8, 698 0, 699 eEncodingIEEE754, 700 eFormatFloat, 701 {LLDB_INVALID_REGNUM, dwarf_d4, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 702 LLDB_INVALID_REGNUM}, 703 nullptr, 704 nullptr, 705 nullptr, 706 0}, 707 {"d5", 708 nullptr, 709 8, 710 0, 711 eEncodingIEEE754, 712 eFormatFloat, 713 {LLDB_INVALID_REGNUM, dwarf_d5, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 714 LLDB_INVALID_REGNUM}, 715 nullptr, 716 nullptr, 717 nullptr, 718 0}, 719 {"d6", 720 nullptr, 721 8, 722 0, 723 eEncodingIEEE754, 724 eFormatFloat, 725 {LLDB_INVALID_REGNUM, dwarf_d6, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 726 LLDB_INVALID_REGNUM}, 727 nullptr, 728 nullptr, 729 nullptr, 730 0}, 731 {"d7", 732 nullptr, 733 8, 734 0, 735 eEncodingIEEE754, 736 eFormatFloat, 737 {LLDB_INVALID_REGNUM, dwarf_d7, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 738 LLDB_INVALID_REGNUM}, 739 nullptr, 740 nullptr, 741 nullptr, 742 0}, 743 {"d8", 744 nullptr, 745 8, 746 0, 747 eEncodingIEEE754, 748 eFormatFloat, 749 {LLDB_INVALID_REGNUM, dwarf_d8, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 750 LLDB_INVALID_REGNUM}, 751 nullptr, 752 nullptr, 753 nullptr, 754 0}, 755 {"d9", 756 nullptr, 757 8, 758 0, 759 eEncodingIEEE754, 760 eFormatFloat, 761 {LLDB_INVALID_REGNUM, dwarf_d9, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 762 LLDB_INVALID_REGNUM}, 763 nullptr, 764 nullptr, 765 nullptr, 766 0}, 767 {"d10", 768 nullptr, 769 8, 770 0, 771 eEncodingIEEE754, 772 eFormatFloat, 773 {LLDB_INVALID_REGNUM, dwarf_d10, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 774 LLDB_INVALID_REGNUM}, 775 nullptr, 776 nullptr, 777 nullptr, 778 0}, 779 {"d11", 780 nullptr, 781 8, 782 0, 783 eEncodingIEEE754, 784 eFormatFloat, 785 {LLDB_INVALID_REGNUM, dwarf_d11, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 786 LLDB_INVALID_REGNUM}, 787 nullptr, 788 nullptr, 789 nullptr, 790 0}, 791 {"d12", 792 nullptr, 793 8, 794 0, 795 eEncodingIEEE754, 796 eFormatFloat, 797 {LLDB_INVALID_REGNUM, dwarf_d12, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 798 LLDB_INVALID_REGNUM}, 799 nullptr, 800 nullptr, 801 nullptr, 802 0}, 803 {"d13", 804 nullptr, 805 8, 806 0, 807 eEncodingIEEE754, 808 eFormatFloat, 809 {LLDB_INVALID_REGNUM, dwarf_d13, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 810 LLDB_INVALID_REGNUM}, 811 nullptr, 812 nullptr, 813 nullptr, 814 0}, 815 {"d14", 816 nullptr, 817 8, 818 0, 819 eEncodingIEEE754, 820 eFormatFloat, 821 {LLDB_INVALID_REGNUM, dwarf_d14, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 822 LLDB_INVALID_REGNUM}, 823 nullptr, 824 nullptr, 825 nullptr, 826 0}, 827 {"d15", 828 nullptr, 829 8, 830 0, 831 eEncodingIEEE754, 832 eFormatFloat, 833 {LLDB_INVALID_REGNUM, dwarf_d15, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 834 LLDB_INVALID_REGNUM}, 835 nullptr, 836 nullptr, 837 nullptr, 838 0}, 839 {"d16", 840 nullptr, 841 8, 842 0, 843 eEncodingIEEE754, 844 eFormatFloat, 845 {LLDB_INVALID_REGNUM, dwarf_d16, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 846 LLDB_INVALID_REGNUM}, 847 nullptr, 848 nullptr, 849 nullptr, 850 0}, 851 {"d17", 852 nullptr, 853 8, 854 0, 855 eEncodingIEEE754, 856 eFormatFloat, 857 {LLDB_INVALID_REGNUM, dwarf_d17, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 858 LLDB_INVALID_REGNUM}, 859 nullptr, 860 nullptr, 861 nullptr, 862 0}, 863 {"d18", 864 nullptr, 865 8, 866 0, 867 eEncodingIEEE754, 868 eFormatFloat, 869 {LLDB_INVALID_REGNUM, dwarf_d18, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 870 LLDB_INVALID_REGNUM}, 871 nullptr, 872 nullptr, 873 nullptr, 874 0}, 875 {"d19", 876 nullptr, 877 8, 878 0, 879 eEncodingIEEE754, 880 eFormatFloat, 881 {LLDB_INVALID_REGNUM, dwarf_d19, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 882 LLDB_INVALID_REGNUM}, 883 nullptr, 884 nullptr, 885 nullptr, 886 0}, 887 {"d20", 888 nullptr, 889 8, 890 0, 891 eEncodingIEEE754, 892 eFormatFloat, 893 {LLDB_INVALID_REGNUM, dwarf_d20, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 894 LLDB_INVALID_REGNUM}, 895 nullptr, 896 nullptr, 897 nullptr, 898 0}, 899 {"d21", 900 nullptr, 901 8, 902 0, 903 eEncodingIEEE754, 904 eFormatFloat, 905 {LLDB_INVALID_REGNUM, dwarf_d21, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 906 LLDB_INVALID_REGNUM}, 907 nullptr, 908 nullptr, 909 nullptr, 910 0}, 911 {"d22", 912 nullptr, 913 8, 914 0, 915 eEncodingIEEE754, 916 eFormatFloat, 917 {LLDB_INVALID_REGNUM, dwarf_d22, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 918 LLDB_INVALID_REGNUM}, 919 nullptr, 920 nullptr, 921 nullptr, 922 0}, 923 {"d23", 924 nullptr, 925 8, 926 0, 927 eEncodingIEEE754, 928 eFormatFloat, 929 {LLDB_INVALID_REGNUM, dwarf_d23, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 930 LLDB_INVALID_REGNUM}, 931 nullptr, 932 nullptr, 933 nullptr, 934 0}, 935 {"d24", 936 nullptr, 937 8, 938 0, 939 eEncodingIEEE754, 940 eFormatFloat, 941 {LLDB_INVALID_REGNUM, dwarf_d24, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 942 LLDB_INVALID_REGNUM}, 943 nullptr, 944 nullptr, 945 nullptr, 946 0}, 947 {"d25", 948 nullptr, 949 8, 950 0, 951 eEncodingIEEE754, 952 eFormatFloat, 953 {LLDB_INVALID_REGNUM, dwarf_d25, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 954 LLDB_INVALID_REGNUM}, 955 nullptr, 956 nullptr, 957 nullptr, 958 0}, 959 {"d26", 960 nullptr, 961 8, 962 0, 963 eEncodingIEEE754, 964 eFormatFloat, 965 {LLDB_INVALID_REGNUM, dwarf_d26, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 966 LLDB_INVALID_REGNUM}, 967 nullptr, 968 nullptr, 969 nullptr, 970 0}, 971 {"d27", 972 nullptr, 973 8, 974 0, 975 eEncodingIEEE754, 976 eFormatFloat, 977 {LLDB_INVALID_REGNUM, dwarf_d27, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 978 LLDB_INVALID_REGNUM}, 979 nullptr, 980 nullptr, 981 nullptr, 982 0}, 983 {"d28", 984 nullptr, 985 8, 986 0, 987 eEncodingIEEE754, 988 eFormatFloat, 989 {LLDB_INVALID_REGNUM, dwarf_d28, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 990 LLDB_INVALID_REGNUM}, 991 nullptr, 992 nullptr, 993 nullptr, 994 0}, 995 {"d29", 996 nullptr, 997 8, 998 0, 999 eEncodingIEEE754, 1000 eFormatFloat, 1001 {LLDB_INVALID_REGNUM, dwarf_d29, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 1002 LLDB_INVALID_REGNUM}, 1003 nullptr, 1004 nullptr, 1005 nullptr, 1006 0}, 1007 {"d30", 1008 nullptr, 1009 8, 1010 0, 1011 eEncodingIEEE754, 1012 eFormatFloat, 1013 {LLDB_INVALID_REGNUM, dwarf_d30, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 1014 LLDB_INVALID_REGNUM}, 1015 nullptr, 1016 nullptr, 1017 nullptr, 1018 0}, 1019 {"d31", 1020 nullptr, 1021 8, 1022 0, 1023 eEncodingIEEE754, 1024 eFormatFloat, 1025 {LLDB_INVALID_REGNUM, dwarf_d31, LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM, 1026 LLDB_INVALID_REGNUM}, 1027 nullptr, 1028 nullptr, 1029 nullptr, 1030 0}, 1031 {"r8_usr", 1032 nullptr, 1033 4, 1034 0, 1035 eEncodingUint, 1036 eFormatHex, 1037 {LLDB_INVALID_REGNUM, dwarf_r8_usr, LLDB_INVALID_REGNUM, 1038 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1039 nullptr, 1040 nullptr, 1041 nullptr, 1042 0}, 1043 {"r9_usr", 1044 nullptr, 1045 4, 1046 0, 1047 eEncodingUint, 1048 eFormatHex, 1049 {LLDB_INVALID_REGNUM, dwarf_r9_usr, LLDB_INVALID_REGNUM, 1050 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1051 nullptr, 1052 nullptr, 1053 nullptr, 1054 0}, 1055 {"r10_usr", 1056 nullptr, 1057 4, 1058 0, 1059 eEncodingUint, 1060 eFormatHex, 1061 {LLDB_INVALID_REGNUM, dwarf_r10_usr, LLDB_INVALID_REGNUM, 1062 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1063 nullptr, 1064 nullptr, 1065 nullptr, 1066 0}, 1067 {"r11_usr", 1068 nullptr, 1069 4, 1070 0, 1071 eEncodingUint, 1072 eFormatHex, 1073 {LLDB_INVALID_REGNUM, dwarf_r11_usr, LLDB_INVALID_REGNUM, 1074 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1075 nullptr, 1076 nullptr, 1077 nullptr, 1078 0}, 1079 {"r12_usr", 1080 nullptr, 1081 4, 1082 0, 1083 eEncodingUint, 1084 eFormatHex, 1085 {LLDB_INVALID_REGNUM, dwarf_r12_usr, LLDB_INVALID_REGNUM, 1086 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1087 nullptr, 1088 nullptr, 1089 nullptr, 1090 0}, 1091 {"r13_usr", 1092 "sp_usr", 1093 4, 1094 0, 1095 eEncodingUint, 1096 eFormatHex, 1097 {LLDB_INVALID_REGNUM, dwarf_r13_usr, LLDB_INVALID_REGNUM, 1098 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1099 nullptr, 1100 nullptr, 1101 nullptr, 1102 0}, 1103 {"r14_usr", 1104 "lr_usr", 1105 4, 1106 0, 1107 eEncodingUint, 1108 eFormatHex, 1109 {LLDB_INVALID_REGNUM, dwarf_r14_usr, LLDB_INVALID_REGNUM, 1110 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1111 nullptr, 1112 nullptr, 1113 nullptr, 1114 0}, 1115 {"r8_fiq", 1116 nullptr, 1117 4, 1118 0, 1119 eEncodingUint, 1120 eFormatHex, 1121 {LLDB_INVALID_REGNUM, dwarf_r8_fiq, LLDB_INVALID_REGNUM, 1122 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1123 nullptr, 1124 nullptr, 1125 nullptr, 1126 0}, 1127 {"r9_fiq", 1128 nullptr, 1129 4, 1130 0, 1131 eEncodingUint, 1132 eFormatHex, 1133 {LLDB_INVALID_REGNUM, dwarf_r9_fiq, LLDB_INVALID_REGNUM, 1134 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1135 nullptr, 1136 nullptr, 1137 nullptr, 1138 0}, 1139 {"r10_fiq", 1140 nullptr, 1141 4, 1142 0, 1143 eEncodingUint, 1144 eFormatHex, 1145 {LLDB_INVALID_REGNUM, dwarf_r10_fiq, LLDB_INVALID_REGNUM, 1146 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1147 nullptr, 1148 nullptr, 1149 nullptr, 1150 0}, 1151 {"r11_fiq", 1152 nullptr, 1153 4, 1154 0, 1155 eEncodingUint, 1156 eFormatHex, 1157 {LLDB_INVALID_REGNUM, dwarf_r11_fiq, LLDB_INVALID_REGNUM, 1158 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1159 nullptr, 1160 nullptr, 1161 nullptr, 1162 0}, 1163 {"r12_fiq", 1164 nullptr, 1165 4, 1166 0, 1167 eEncodingUint, 1168 eFormatHex, 1169 {LLDB_INVALID_REGNUM, dwarf_r12_fiq, LLDB_INVALID_REGNUM, 1170 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1171 nullptr, 1172 nullptr, 1173 nullptr, 1174 0}, 1175 {"r13_fiq", 1176 "sp_fiq", 1177 4, 1178 0, 1179 eEncodingUint, 1180 eFormatHex, 1181 {LLDB_INVALID_REGNUM, dwarf_r13_fiq, LLDB_INVALID_REGNUM, 1182 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1183 nullptr, 1184 nullptr, 1185 nullptr, 1186 0}, 1187 {"r14_fiq", 1188 "lr_fiq", 1189 4, 1190 0, 1191 eEncodingUint, 1192 eFormatHex, 1193 {LLDB_INVALID_REGNUM, dwarf_r14_fiq, LLDB_INVALID_REGNUM, 1194 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1195 nullptr, 1196 nullptr, 1197 nullptr, 1198 0}, 1199 {"r13_irq", 1200 "sp_irq", 1201 4, 1202 0, 1203 eEncodingUint, 1204 eFormatHex, 1205 {LLDB_INVALID_REGNUM, dwarf_r13_irq, LLDB_INVALID_REGNUM, 1206 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1207 nullptr, 1208 nullptr, 1209 nullptr, 1210 0}, 1211 {"r14_irq", 1212 "lr_irq", 1213 4, 1214 0, 1215 eEncodingUint, 1216 eFormatHex, 1217 {LLDB_INVALID_REGNUM, dwarf_r14_irq, LLDB_INVALID_REGNUM, 1218 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1219 nullptr, 1220 nullptr, 1221 nullptr, 1222 0}, 1223 {"r13_abt", 1224 "sp_abt", 1225 4, 1226 0, 1227 eEncodingUint, 1228 eFormatHex, 1229 {LLDB_INVALID_REGNUM, dwarf_r13_abt, LLDB_INVALID_REGNUM, 1230 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1231 nullptr, 1232 nullptr, 1233 nullptr, 1234 0}, 1235 {"r14_abt", 1236 "lr_abt", 1237 4, 1238 0, 1239 eEncodingUint, 1240 eFormatHex, 1241 {LLDB_INVALID_REGNUM, dwarf_r14_abt, LLDB_INVALID_REGNUM, 1242 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1243 nullptr, 1244 nullptr, 1245 nullptr, 1246 0}, 1247 {"r13_und", 1248 "sp_und", 1249 4, 1250 0, 1251 eEncodingUint, 1252 eFormatHex, 1253 {LLDB_INVALID_REGNUM, dwarf_r13_und, LLDB_INVALID_REGNUM, 1254 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1255 nullptr, 1256 nullptr, 1257 nullptr, 1258 0}, 1259 {"r14_und", 1260 "lr_und", 1261 4, 1262 0, 1263 eEncodingUint, 1264 eFormatHex, 1265 {LLDB_INVALID_REGNUM, dwarf_r14_und, LLDB_INVALID_REGNUM, 1266 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1267 nullptr, 1268 nullptr, 1269 nullptr, 1270 0}, 1271 {"r13_svc", 1272 "sp_svc", 1273 4, 1274 0, 1275 eEncodingUint, 1276 eFormatHex, 1277 {LLDB_INVALID_REGNUM, dwarf_r13_svc, LLDB_INVALID_REGNUM, 1278 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1279 nullptr, 1280 nullptr, 1281 nullptr, 1282 0}, 1283 {"r14_svc", 1284 "lr_svc", 1285 4, 1286 0, 1287 eEncodingUint, 1288 eFormatHex, 1289 {LLDB_INVALID_REGNUM, dwarf_r14_svc, LLDB_INVALID_REGNUM, 1290 LLDB_INVALID_REGNUM, LLDB_INVALID_REGNUM}, 1291 nullptr, 1292 nullptr, 1293 nullptr, 1294 0}}; 1295 1296 static const uint32_t k_num_register_infos = 1297 llvm::array_lengthof(g_register_infos); 1298 1299 const lldb_private::RegisterInfo * 1300 ABISysV_arm::GetRegisterInfoArray(uint32_t &count) { 1301 count = k_num_register_infos; 1302 return g_register_infos; 1303 } 1304 1305 size_t ABISysV_arm::GetRedZoneSize() const { return 0; } 1306 1307 // Static Functions 1308 1309 ABISP 1310 ABISysV_arm::CreateInstance(lldb::ProcessSP process_sp, const ArchSpec &arch) { 1311 const llvm::Triple::ArchType arch_type = arch.GetTriple().getArch(); 1312 const llvm::Triple::VendorType vendor_type = arch.GetTriple().getVendor(); 1313 1314 if (vendor_type != llvm::Triple::Apple) { 1315 if ((arch_type == llvm::Triple::arm) || 1316 (arch_type == llvm::Triple::thumb)) { 1317 return ABISP( 1318 new ABISysV_arm(std::move(process_sp), MakeMCRegisterInfo(arch))); 1319 } 1320 } 1321 1322 return ABISP(); 1323 } 1324 1325 bool ABISysV_arm::PrepareTrivialCall(Thread &thread, addr_t sp, 1326 addr_t function_addr, addr_t return_addr, 1327 llvm::ArrayRef<addr_t> args) const { 1328 RegisterContext *reg_ctx = thread.GetRegisterContext().get(); 1329 if (!reg_ctx) 1330 return false; 1331 1332 const uint32_t pc_reg_num = reg_ctx->ConvertRegisterKindToRegisterNumber( 1333 eRegisterKindGeneric, LLDB_REGNUM_GENERIC_PC); 1334 const uint32_t sp_reg_num = reg_ctx->ConvertRegisterKindToRegisterNumber( 1335 eRegisterKindGeneric, LLDB_REGNUM_GENERIC_SP); 1336 const uint32_t ra_reg_num = reg_ctx->ConvertRegisterKindToRegisterNumber( 1337 eRegisterKindGeneric, LLDB_REGNUM_GENERIC_RA); 1338 1339 RegisterValue reg_value; 1340 1341 const uint8_t reg_names[] = { 1342 LLDB_REGNUM_GENERIC_ARG1, LLDB_REGNUM_GENERIC_ARG2, 1343 LLDB_REGNUM_GENERIC_ARG3, LLDB_REGNUM_GENERIC_ARG4}; 1344 1345 llvm::ArrayRef<addr_t>::iterator ai = args.begin(), ae = args.end(); 1346 1347 for (size_t i = 0; i < llvm::array_lengthof(reg_names); ++i) { 1348 if (ai == ae) 1349 break; 1350 1351 reg_value.SetUInt32(*ai); 1352 if (!reg_ctx->WriteRegister( 1353 reg_ctx->GetRegisterInfo(eRegisterKindGeneric, reg_names[i]), 1354 reg_value)) 1355 return false; 1356 1357 ++ai; 1358 } 1359 1360 if (ai != ae) { 1361 // Spill onto the stack 1362 size_t num_stack_regs = ae - ai; 1363 1364 sp -= (num_stack_regs * 4); 1365 // Keep the stack 8 byte aligned, not that we need to 1366 sp &= ~(8ull - 1ull); 1367 1368 // just using arg1 to get the right size 1369 const RegisterInfo *reg_info = reg_ctx->GetRegisterInfo( 1370 eRegisterKindGeneric, LLDB_REGNUM_GENERIC_ARG1); 1371 1372 addr_t arg_pos = sp; 1373 1374 for (; ai != ae; ++ai) { 1375 reg_value.SetUInt32(*ai); 1376 if (reg_ctx 1377 ->WriteRegisterValueToMemory(reg_info, arg_pos, 1378 reg_info->byte_size, reg_value) 1379 .Fail()) 1380 return false; 1381 arg_pos += reg_info->byte_size; 1382 } 1383 } 1384 1385 TargetSP target_sp(thread.CalculateTarget()); 1386 Address so_addr; 1387 1388 // Figure out if our return address is ARM or Thumb by using the 1389 // Address::GetCallableLoadAddress(Target*) which will figure out the ARM 1390 // thumb-ness and set the correct address bits for us. 1391 so_addr.SetLoadAddress(return_addr, target_sp.get()); 1392 return_addr = so_addr.GetCallableLoadAddress(target_sp.get()); 1393 1394 // Set "lr" to the return address 1395 if (!reg_ctx->WriteRegisterFromUnsigned(ra_reg_num, return_addr)) 1396 return false; 1397 1398 // Set "sp" to the requested value 1399 if (!reg_ctx->WriteRegisterFromUnsigned(sp_reg_num, sp)) 1400 return false; 1401 1402 // If bit zero or 1 is set, this must be a thumb function, no need to figure 1403 // this out from the symbols. 1404 so_addr.SetLoadAddress(function_addr, target_sp.get()); 1405 function_addr = so_addr.GetCallableLoadAddress(target_sp.get()); 1406 1407 const RegisterInfo *cpsr_reg_info = 1408 reg_ctx->GetRegisterInfo(eRegisterKindGeneric, LLDB_REGNUM_GENERIC_FLAGS); 1409 const uint32_t curr_cpsr = reg_ctx->ReadRegisterAsUnsigned(cpsr_reg_info, 0); 1410 1411 // Make a new CPSR and mask out any Thumb IT (if/then) bits 1412 uint32_t new_cpsr = curr_cpsr & ~MASK_CPSR_IT_MASK; 1413 // If bit zero or 1 is set, this must be thumb... 1414 if (function_addr & 1ull) 1415 new_cpsr |= MASK_CPSR_T; // Set T bit in CPSR 1416 else 1417 new_cpsr &= ~MASK_CPSR_T; // Clear T bit in CPSR 1418 1419 if (new_cpsr != curr_cpsr) { 1420 if (!reg_ctx->WriteRegisterFromUnsigned(cpsr_reg_info, new_cpsr)) 1421 return false; 1422 } 1423 1424 function_addr &= 1425 ~1ull; // clear bit zero since the CPSR will take care of the mode for us 1426 1427 // Set "pc" to the address requested 1428 return reg_ctx->WriteRegisterFromUnsigned(pc_reg_num, function_addr); 1429 } 1430 1431 bool ABISysV_arm::GetArgumentValues(Thread &thread, ValueList &values) const { 1432 uint32_t num_values = values.GetSize(); 1433 1434 ExecutionContext exe_ctx(thread.shared_from_this()); 1435 // For now, assume that the types in the AST values come from the Target's 1436 // scratch AST. 1437 1438 // Extract the register context so we can read arguments from registers 1439 1440 RegisterContext *reg_ctx = thread.GetRegisterContext().get(); 1441 1442 if (!reg_ctx) 1443 return false; 1444 1445 addr_t sp = 0; 1446 1447 for (uint32_t value_idx = 0; value_idx < num_values; ++value_idx) { 1448 // We currently only support extracting values with Clang QualTypes. Do we 1449 // care about others? 1450 Value *value = values.GetValueAtIndex(value_idx); 1451 1452 if (!value) 1453 return false; 1454 1455 CompilerType compiler_type = value->GetCompilerType(); 1456 if (compiler_type) { 1457 bool is_signed = false; 1458 size_t bit_width = 0; 1459 if (compiler_type.IsIntegerOrEnumerationType(is_signed) || 1460 compiler_type.IsPointerOrReferenceType()) { 1461 if (llvm::Optional<uint64_t> size = compiler_type.GetBitSize(&thread)) 1462 bit_width = *size; 1463 } else { 1464 // We only handle integer, pointer and reference types currently... 1465 return false; 1466 } 1467 1468 if (bit_width <= (exe_ctx.GetProcessRef().GetAddressByteSize() * 8)) { 1469 if (value_idx < 4) { 1470 // Arguments 1-4 are in r0-r3... 1471 const RegisterInfo *arg_reg_info = nullptr; 1472 arg_reg_info = reg_ctx->GetRegisterInfo( 1473 eRegisterKindGeneric, LLDB_REGNUM_GENERIC_ARG1 + value_idx); 1474 if (arg_reg_info) { 1475 RegisterValue reg_value; 1476 1477 if (reg_ctx->ReadRegister(arg_reg_info, reg_value)) { 1478 if (is_signed) 1479 reg_value.SignExtend(bit_width); 1480 if (!reg_value.GetScalarValue(value->GetScalar())) 1481 return false; 1482 continue; 1483 } 1484 } 1485 return false; 1486 } else { 1487 if (sp == 0) { 1488 // Read the stack pointer if it already hasn't been read 1489 sp = reg_ctx->GetSP(0); 1490 if (sp == 0) 1491 return false; 1492 } 1493 1494 // Arguments 5 on up are on the stack 1495 const uint32_t arg_byte_size = (bit_width + (8 - 1)) / 8; 1496 Status error; 1497 if (!exe_ctx.GetProcessRef().ReadScalarIntegerFromMemory( 1498 sp, arg_byte_size, is_signed, value->GetScalar(), error)) 1499 return false; 1500 1501 sp += arg_byte_size; 1502 } 1503 } 1504 } 1505 } 1506 return true; 1507 } 1508 1509 static bool GetReturnValuePassedInMemory(Thread &thread, 1510 RegisterContext *reg_ctx, 1511 size_t byte_size, Value &value) { 1512 Status error; 1513 DataBufferHeap buffer(byte_size, 0); 1514 1515 const RegisterInfo *r0_reg_info = 1516 reg_ctx->GetRegisterInfo(eRegisterKindGeneric, LLDB_REGNUM_GENERIC_ARG1); 1517 uint32_t address = 1518 reg_ctx->ReadRegisterAsUnsigned(r0_reg_info, 0) & UINT32_MAX; 1519 thread.GetProcess()->ReadMemory(address, buffer.GetBytes(), 1520 buffer.GetByteSize(), error); 1521 1522 if (error.Fail()) 1523 return false; 1524 1525 value.SetBytes(buffer.GetBytes(), buffer.GetByteSize()); 1526 return true; 1527 } 1528 1529 bool ABISysV_arm::IsArmHardFloat(Thread &thread) const { 1530 ProcessSP process_sp(thread.GetProcess()); 1531 if (process_sp) { 1532 const ArchSpec &arch(process_sp->GetTarget().GetArchitecture()); 1533 1534 return (arch.GetFlags() & ArchSpec::eARM_abi_hard_float) != 0; 1535 } 1536 1537 return false; 1538 } 1539 1540 ValueObjectSP ABISysV_arm::GetReturnValueObjectImpl( 1541 Thread &thread, lldb_private::CompilerType &compiler_type) const { 1542 Value value; 1543 ValueObjectSP return_valobj_sp; 1544 1545 if (!compiler_type) 1546 return return_valobj_sp; 1547 1548 // value.SetContext (Value::eContextTypeClangType, 1549 // compiler_type.GetOpaqueQualType()); 1550 value.SetCompilerType(compiler_type); 1551 1552 RegisterContext *reg_ctx = thread.GetRegisterContext().get(); 1553 if (!reg_ctx) 1554 return return_valobj_sp; 1555 1556 bool is_signed; 1557 bool is_complex; 1558 uint32_t float_count; 1559 bool is_vfp_candidate = false; 1560 uint8_t vfp_count = 0; 1561 uint8_t vfp_byte_size = 0; 1562 1563 // Get the pointer to the first stack argument so we have a place to start 1564 // when reading data 1565 1566 const RegisterInfo *r0_reg_info = 1567 reg_ctx->GetRegisterInfo(eRegisterKindGeneric, LLDB_REGNUM_GENERIC_ARG1); 1568 llvm::Optional<uint64_t> bit_width = compiler_type.GetBitSize(&thread); 1569 llvm::Optional<uint64_t> byte_size = compiler_type.GetByteSize(&thread); 1570 if (!bit_width || !byte_size) 1571 return return_valobj_sp; 1572 1573 if (compiler_type.IsIntegerOrEnumerationType(is_signed)) { 1574 switch (*bit_width) { 1575 default: 1576 return return_valobj_sp; 1577 case 64: { 1578 const RegisterInfo *r1_reg_info = reg_ctx->GetRegisterInfo( 1579 eRegisterKindGeneric, LLDB_REGNUM_GENERIC_ARG2); 1580 uint64_t raw_value; 1581 raw_value = reg_ctx->ReadRegisterAsUnsigned(r0_reg_info, 0) & UINT32_MAX; 1582 raw_value |= ((uint64_t)(reg_ctx->ReadRegisterAsUnsigned(r1_reg_info, 0) & 1583 UINT32_MAX)) 1584 << 32; 1585 if (is_signed) 1586 value.GetScalar() = (int64_t)raw_value; 1587 else 1588 value.GetScalar() = (uint64_t)raw_value; 1589 } break; 1590 case 32: 1591 if (is_signed) 1592 value.GetScalar() = (int32_t)( 1593 reg_ctx->ReadRegisterAsUnsigned(r0_reg_info, 0) & UINT32_MAX); 1594 else 1595 value.GetScalar() = (uint32_t)( 1596 reg_ctx->ReadRegisterAsUnsigned(r0_reg_info, 0) & UINT32_MAX); 1597 break; 1598 case 16: 1599 if (is_signed) 1600 value.GetScalar() = (int16_t)( 1601 reg_ctx->ReadRegisterAsUnsigned(r0_reg_info, 0) & UINT16_MAX); 1602 else 1603 value.GetScalar() = (uint16_t)( 1604 reg_ctx->ReadRegisterAsUnsigned(r0_reg_info, 0) & UINT16_MAX); 1605 break; 1606 case 8: 1607 if (is_signed) 1608 value.GetScalar() = (int8_t)( 1609 reg_ctx->ReadRegisterAsUnsigned(r0_reg_info, 0) & UINT8_MAX); 1610 else 1611 value.GetScalar() = (uint8_t)( 1612 reg_ctx->ReadRegisterAsUnsigned(r0_reg_info, 0) & UINT8_MAX); 1613 break; 1614 } 1615 } else if (compiler_type.IsPointerType()) { 1616 uint32_t ptr = 1617 thread.GetRegisterContext()->ReadRegisterAsUnsigned(r0_reg_info, 0) & 1618 UINT32_MAX; 1619 value.GetScalar() = ptr; 1620 } else if (compiler_type.IsVectorType(nullptr, nullptr)) { 1621 if (IsArmHardFloat(thread) && (*byte_size == 8 || *byte_size == 16)) { 1622 is_vfp_candidate = true; 1623 vfp_byte_size = 8; 1624 vfp_count = (*byte_size == 8 ? 1 : 2); 1625 } else if (*byte_size <= 16) { 1626 DataBufferHeap buffer(16, 0); 1627 uint32_t *buffer_ptr = (uint32_t *)buffer.GetBytes(); 1628 1629 for (uint32_t i = 0; 4 * i < *byte_size; ++i) { 1630 const RegisterInfo *reg_info = reg_ctx->GetRegisterInfo( 1631 eRegisterKindGeneric, LLDB_REGNUM_GENERIC_ARG1 + i); 1632 buffer_ptr[i] = 1633 reg_ctx->ReadRegisterAsUnsigned(reg_info, 0) & UINT32_MAX; 1634 } 1635 value.SetBytes(buffer.GetBytes(), *byte_size); 1636 } else { 1637 if (!GetReturnValuePassedInMemory(thread, reg_ctx, *byte_size, value)) 1638 return return_valobj_sp; 1639 } 1640 } else if (compiler_type.IsFloatingPointType(float_count, is_complex)) { 1641 if (float_count == 1 && !is_complex) { 1642 switch (*bit_width) { 1643 default: 1644 return return_valobj_sp; 1645 case 64: { 1646 static_assert(sizeof(double) == sizeof(uint64_t), ""); 1647 1648 if (IsArmHardFloat(thread)) { 1649 RegisterValue reg_value; 1650 const RegisterInfo *d0_reg_info = 1651 reg_ctx->GetRegisterInfoByName("d0", 0); 1652 reg_ctx->ReadRegister(d0_reg_info, reg_value); 1653 value.GetScalar() = reg_value.GetAsDouble(); 1654 } else { 1655 uint64_t raw_value; 1656 const RegisterInfo *r1_reg_info = reg_ctx->GetRegisterInfo( 1657 eRegisterKindGeneric, LLDB_REGNUM_GENERIC_ARG2); 1658 raw_value = 1659 reg_ctx->ReadRegisterAsUnsigned(r0_reg_info, 0) & UINT32_MAX; 1660 raw_value |= 1661 ((uint64_t)(reg_ctx->ReadRegisterAsUnsigned(r1_reg_info, 0) & 1662 UINT32_MAX)) 1663 << 32; 1664 value.GetScalar() = *reinterpret_cast<double *>(&raw_value); 1665 } 1666 break; 1667 } 1668 case 16: // Half precision returned after a conversion to single precision 1669 case 32: { 1670 static_assert(sizeof(float) == sizeof(uint32_t), ""); 1671 1672 if (IsArmHardFloat(thread)) { 1673 RegisterValue reg_value; 1674 const RegisterInfo *s0_reg_info = 1675 reg_ctx->GetRegisterInfoByName("s0", 0); 1676 reg_ctx->ReadRegister(s0_reg_info, reg_value); 1677 value.GetScalar() = reg_value.GetAsFloat(); 1678 } else { 1679 uint32_t raw_value; 1680 raw_value = 1681 reg_ctx->ReadRegisterAsUnsigned(r0_reg_info, 0) & UINT32_MAX; 1682 value.GetScalar() = *reinterpret_cast<float *>(&raw_value); 1683 } 1684 break; 1685 } 1686 } 1687 } else if (is_complex && float_count == 2) { 1688 if (IsArmHardFloat(thread)) { 1689 is_vfp_candidate = true; 1690 vfp_byte_size = *byte_size / 2; 1691 vfp_count = 2; 1692 } else if (!GetReturnValuePassedInMemory(thread, reg_ctx, *bit_width / 8, 1693 value)) 1694 return return_valobj_sp; 1695 } else 1696 // not handled yet 1697 return return_valobj_sp; 1698 } else if (compiler_type.IsAggregateType()) { 1699 if (IsArmHardFloat(thread)) { 1700 CompilerType base_type; 1701 const uint32_t homogeneous_count = 1702 compiler_type.IsHomogeneousAggregate(&base_type); 1703 1704 if (homogeneous_count > 0 && homogeneous_count <= 4) { 1705 llvm::Optional<uint64_t> base_byte_size = 1706 base_type.GetByteSize(&thread); 1707 if (base_type.IsVectorType(nullptr, nullptr)) { 1708 if (base_byte_size && 1709 (*base_byte_size == 8 || *base_byte_size == 16)) { 1710 is_vfp_candidate = true; 1711 vfp_byte_size = 8; 1712 vfp_count = (*base_byte_size == 8 ? homogeneous_count 1713 : homogeneous_count * 2); 1714 } 1715 } else if (base_type.IsFloatingPointType(float_count, is_complex)) { 1716 if (float_count == 1 && !is_complex) { 1717 is_vfp_candidate = true; 1718 if (base_byte_size) 1719 vfp_byte_size = *base_byte_size; 1720 vfp_count = homogeneous_count; 1721 } 1722 } 1723 } else if (homogeneous_count == 0) { 1724 const uint32_t num_children = compiler_type.GetNumFields(); 1725 1726 if (num_children > 0 && num_children <= 2) { 1727 uint32_t index = 0; 1728 for (index = 0; index < num_children; index++) { 1729 std::string name; 1730 base_type = compiler_type.GetFieldAtIndex(index, name, nullptr, 1731 nullptr, nullptr); 1732 1733 if (base_type.IsFloatingPointType(float_count, is_complex)) { 1734 llvm::Optional<uint64_t> base_byte_size = 1735 base_type.GetByteSize(&thread); 1736 if (float_count == 2 && is_complex) { 1737 if (index != 0 && base_byte_size && 1738 vfp_byte_size != *base_byte_size) 1739 break; 1740 else if (base_byte_size) 1741 vfp_byte_size = *base_byte_size; 1742 } else 1743 break; 1744 } else 1745 break; 1746 } 1747 1748 if (index == num_children) { 1749 is_vfp_candidate = true; 1750 vfp_byte_size = (vfp_byte_size >> 1); 1751 vfp_count = (num_children << 1); 1752 } 1753 } 1754 } 1755 } 1756 1757 if (*byte_size <= 4) { 1758 RegisterValue r0_reg_value; 1759 uint32_t raw_value = 1760 reg_ctx->ReadRegisterAsUnsigned(r0_reg_info, 0) & UINT32_MAX; 1761 value.SetBytes(&raw_value, *byte_size); 1762 } else if (!is_vfp_candidate) { 1763 if (!GetReturnValuePassedInMemory(thread, reg_ctx, *byte_size, value)) 1764 return return_valobj_sp; 1765 } 1766 } else { 1767 // not handled yet 1768 return return_valobj_sp; 1769 } 1770 1771 if (is_vfp_candidate) { 1772 ProcessSP process_sp(thread.GetProcess()); 1773 ByteOrder byte_order = process_sp->GetByteOrder(); 1774 1775 DataBufferSP data_sp(new DataBufferHeap(*byte_size, 0)); 1776 uint32_t data_offset = 0; 1777 1778 for (uint32_t reg_index = 0; reg_index < vfp_count; reg_index++) { 1779 uint32_t regnum = 0; 1780 1781 if (vfp_byte_size == 4) 1782 regnum = dwarf_s0 + reg_index; 1783 else if (vfp_byte_size == 8) 1784 regnum = dwarf_d0 + reg_index; 1785 else 1786 break; 1787 1788 const RegisterInfo *reg_info = 1789 reg_ctx->GetRegisterInfo(eRegisterKindDWARF, regnum); 1790 if (reg_info == nullptr) 1791 break; 1792 1793 RegisterValue reg_value; 1794 if (!reg_ctx->ReadRegister(reg_info, reg_value)) 1795 break; 1796 1797 // Make sure we have enough room in "data_sp" 1798 if ((data_offset + vfp_byte_size) <= data_sp->GetByteSize()) { 1799 Status error; 1800 const size_t bytes_copied = reg_value.GetAsMemoryData( 1801 reg_info, data_sp->GetBytes() + data_offset, vfp_byte_size, 1802 byte_order, error); 1803 if (bytes_copied != vfp_byte_size) 1804 break; 1805 1806 data_offset += bytes_copied; 1807 } 1808 } 1809 1810 if (data_offset == *byte_size) { 1811 DataExtractor data; 1812 data.SetByteOrder(byte_order); 1813 data.SetAddressByteSize(process_sp->GetAddressByteSize()); 1814 data.SetData(data_sp); 1815 1816 return ValueObjectConstResult::Create(&thread, compiler_type, 1817 ConstString(""), data); 1818 } else { // Some error occurred while getting values from registers 1819 return return_valobj_sp; 1820 } 1821 } 1822 1823 // If we get here, we have a valid Value, so make our ValueObject out of it: 1824 1825 return_valobj_sp = ValueObjectConstResult::Create( 1826 thread.GetStackFrameAtIndex(0).get(), value, ConstString("")); 1827 return return_valobj_sp; 1828 } 1829 1830 Status ABISysV_arm::SetReturnValueObject(lldb::StackFrameSP &frame_sp, 1831 lldb::ValueObjectSP &new_value_sp) { 1832 Status error; 1833 if (!new_value_sp) { 1834 error.SetErrorString("Empty value object for return value."); 1835 return error; 1836 } 1837 1838 CompilerType compiler_type = new_value_sp->GetCompilerType(); 1839 if (!compiler_type) { 1840 error.SetErrorString("Null clang type for return value."); 1841 return error; 1842 } 1843 1844 Thread *thread = frame_sp->GetThread().get(); 1845 1846 bool is_signed; 1847 uint32_t count; 1848 bool is_complex; 1849 1850 RegisterContext *reg_ctx = thread->GetRegisterContext().get(); 1851 1852 bool set_it_simple = false; 1853 if (compiler_type.IsIntegerOrEnumerationType(is_signed) || 1854 compiler_type.IsPointerType()) { 1855 DataExtractor data; 1856 Status data_error; 1857 size_t num_bytes = new_value_sp->GetData(data, data_error); 1858 if (data_error.Fail()) { 1859 error.SetErrorStringWithFormat( 1860 "Couldn't convert return value to raw data: %s", 1861 data_error.AsCString()); 1862 return error; 1863 } 1864 lldb::offset_t offset = 0; 1865 if (num_bytes <= 8) { 1866 const RegisterInfo *r0_info = reg_ctx->GetRegisterInfo( 1867 eRegisterKindGeneric, LLDB_REGNUM_GENERIC_ARG1); 1868 if (num_bytes <= 4) { 1869 uint32_t raw_value = data.GetMaxU32(&offset, num_bytes); 1870 1871 if (reg_ctx->WriteRegisterFromUnsigned(r0_info, raw_value)) 1872 set_it_simple = true; 1873 } else { 1874 uint32_t raw_value = data.GetMaxU32(&offset, 4); 1875 1876 if (reg_ctx->WriteRegisterFromUnsigned(r0_info, raw_value)) { 1877 const RegisterInfo *r1_info = reg_ctx->GetRegisterInfo( 1878 eRegisterKindGeneric, LLDB_REGNUM_GENERIC_ARG2); 1879 uint32_t raw_value = data.GetMaxU32(&offset, num_bytes - offset); 1880 1881 if (reg_ctx->WriteRegisterFromUnsigned(r1_info, raw_value)) 1882 set_it_simple = true; 1883 } 1884 } 1885 } else { 1886 error.SetErrorString("We don't support returning longer than 64 bit " 1887 "integer values at present."); 1888 } 1889 } else if (compiler_type.IsFloatingPointType(count, is_complex)) { 1890 if (is_complex) 1891 error.SetErrorString( 1892 "We don't support returning complex values at present"); 1893 else 1894 error.SetErrorString( 1895 "We don't support returning float values at present"); 1896 } 1897 1898 if (!set_it_simple) 1899 error.SetErrorString( 1900 "We only support setting simple integer return types at present."); 1901 1902 return error; 1903 } 1904 1905 bool ABISysV_arm::CreateFunctionEntryUnwindPlan(UnwindPlan &unwind_plan) { 1906 unwind_plan.Clear(); 1907 unwind_plan.SetRegisterKind(eRegisterKindDWARF); 1908 1909 uint32_t lr_reg_num = dwarf_lr; 1910 uint32_t sp_reg_num = dwarf_sp; 1911 uint32_t pc_reg_num = dwarf_pc; 1912 1913 UnwindPlan::RowSP row(new UnwindPlan::Row); 1914 1915 // Our Call Frame Address is the stack pointer value 1916 row->GetCFAValue().SetIsRegisterPlusOffset(sp_reg_num, 0); 1917 1918 // The previous PC is in the LR 1919 row->SetRegisterLocationToRegister(pc_reg_num, lr_reg_num, true); 1920 unwind_plan.AppendRow(row); 1921 1922 // All other registers are the same. 1923 1924 unwind_plan.SetSourceName("arm at-func-entry default"); 1925 unwind_plan.SetSourcedFromCompiler(eLazyBoolNo); 1926 1927 return true; 1928 } 1929 1930 bool ABISysV_arm::CreateDefaultUnwindPlan(UnwindPlan &unwind_plan) { 1931 unwind_plan.Clear(); 1932 unwind_plan.SetRegisterKind(eRegisterKindDWARF); 1933 1934 // TODO: Handle thumb 1935 uint32_t fp_reg_num = dwarf_r11; 1936 uint32_t pc_reg_num = dwarf_pc; 1937 1938 UnwindPlan::RowSP row(new UnwindPlan::Row); 1939 const int32_t ptr_size = 4; 1940 1941 row->GetCFAValue().SetIsRegisterPlusOffset(fp_reg_num, 2 * ptr_size); 1942 row->SetOffset(0); 1943 1944 row->SetRegisterLocationToAtCFAPlusOffset(fp_reg_num, ptr_size * -2, true); 1945 row->SetRegisterLocationToAtCFAPlusOffset(pc_reg_num, ptr_size * -1, true); 1946 1947 unwind_plan.AppendRow(row); 1948 unwind_plan.SetSourceName("arm default unwind plan"); 1949 unwind_plan.SetSourcedFromCompiler(eLazyBoolNo); 1950 unwind_plan.SetUnwindPlanValidAtAllInstructions(eLazyBoolNo); 1951 unwind_plan.SetUnwindPlanForSignalTrap(eLazyBoolNo); 1952 1953 return true; 1954 } 1955 1956 // cf. "ARMv6 Function Calling Conventions" 1957 1958 // ARMv7 on GNU/Linux general purpose reg rules: 1959 // r0-r3 not preserved (used for argument passing) 1960 // r4-r11 preserved (v1-v8) 1961 // r12 not presrved 1962 // r13 preserved (stack pointer) 1963 // r14 preserved (link register) 1964 // r15 preserved (pc) 1965 // cpsr not preserved (different rules for different bits) 1966 1967 // ARMv7 VFP register rules: 1968 // d0-d7 not preserved (aka s0-s15, q0-q3) 1969 // d8-d15 preserved (aka s16-s31, q4-q7) 1970 // d16-d31 not preserved (aka q8-q15) 1971 1972 bool ABISysV_arm::RegisterIsVolatile(const RegisterInfo *reg_info) { 1973 if (reg_info) { 1974 // Volatile registers are: r0, r1, r2, r3, r9, r12, r13 (aka sp) 1975 const char *name = reg_info->name; 1976 if (name[0] == 'r') { 1977 switch (name[1]) { 1978 case '0': 1979 return name[2] == '\0'; // r0 1980 case '1': 1981 switch (name[2]) { 1982 case '\0': 1983 return true; // r1 1984 case '2': 1985 return name[3] == '\0'; // r12 1986 default: 1987 break; 1988 } 1989 break; 1990 1991 case '2': 1992 return name[2] == '\0'; // r2 1993 case '3': 1994 return name[2] == '\0'; // r3 1995 default: 1996 break; 1997 } 1998 } else if (name[0] == 'd') { 1999 switch (name[1]) { 2000 case '0': 2001 return name[2] == '\0'; // d0 is volatile 2002 2003 case '1': 2004 switch (name[2]) { 2005 case '\0': 2006 return true; // d1 is volatile 2007 case '6': 2008 case '7': 2009 case '8': 2010 case '9': 2011 return name[3] == '\0'; // d16 - d19 are volatile 2012 default: 2013 break; 2014 } 2015 break; 2016 2017 case '2': 2018 switch (name[2]) { 2019 case '\0': 2020 return true; // d2 is volatile 2021 case '0': 2022 case '1': 2023 case '2': 2024 case '3': 2025 case '4': 2026 case '5': 2027 case '6': 2028 case '7': 2029 case '8': 2030 case '9': 2031 return name[3] == '\0'; // d20 - d29 are volatile 2032 default: 2033 break; 2034 } 2035 break; 2036 2037 case '3': 2038 switch (name[2]) { 2039 case '\0': 2040 return true; // d3 is volatile 2041 case '0': 2042 case '1': 2043 return name[3] == '\0'; // d30 - d31 are volatile 2044 default: 2045 break; 2046 } 2047 break; 2048 case '4': 2049 case '5': 2050 case '6': 2051 case '7': 2052 return name[2] == '\0'; // d4 - d7 are volatile 2053 2054 default: 2055 break; 2056 } 2057 } else if (name[0] == 's') { 2058 switch (name[1]) { 2059 case '0': 2060 return name[2] == '\0'; // s0 is volatile 2061 2062 case '1': 2063 switch (name[2]) { 2064 case '\0': 2065 return true; // s1 is volatile 2066 case '0': 2067 case '1': 2068 case '2': 2069 case '3': 2070 case '4': 2071 case '5': 2072 return name[3] == '\0'; // s10 - s15 are volatile 2073 default: 2074 break; 2075 } 2076 break; 2077 2078 case '2': 2079 case '3': 2080 case '4': 2081 case '5': 2082 case '6': 2083 case '7': 2084 case '8': 2085 case '9': 2086 return name[2] == '\0'; // s2 - s9 are volatile 2087 2088 default: 2089 break; 2090 } 2091 } else if (name[0] == 'q') { 2092 switch (name[1]) { 2093 case '1': 2094 switch (name[2]) { 2095 case '\0': 2096 return true; // q1 is volatile 2097 case '0': 2098 case '1': 2099 case '2': 2100 case '3': 2101 case '4': 2102 case '5': 2103 return true; // q10-q15 are volatile 2104 default: 2105 return false; 2106 } 2107 break; 2108 2109 case '0': 2110 case '2': 2111 case '3': 2112 return name[2] == '\0'; // q0-q3 are volatile 2113 case '8': 2114 case '9': 2115 return name[2] == '\0'; // q8-q9 are volatile 2116 default: 2117 break; 2118 } 2119 } else if (name[0] == 's' && name[1] == 'p' && name[2] == '\0') 2120 return true; 2121 } 2122 return false; 2123 } 2124 2125 void ABISysV_arm::Initialize() { 2126 PluginManager::RegisterPlugin(GetPluginNameStatic(), 2127 "SysV ABI for arm targets", CreateInstance); 2128 } 2129 2130 void ABISysV_arm::Terminate() { 2131 PluginManager::UnregisterPlugin(CreateInstance); 2132 } 2133 2134 lldb_private::ConstString ABISysV_arm::GetPluginNameStatic() { 2135 static ConstString g_name("SysV-arm"); 2136 return g_name; 2137 } 2138 2139 // PluginInterface protocol 2140 2141 lldb_private::ConstString ABISysV_arm::GetPluginName() { 2142 return GetPluginNameStatic(); 2143 } 2144 2145 uint32_t ABISysV_arm::GetPluginVersion() { return 1; } 2146