1 2/*--------------------------------------------------------------------*/ 3/*--- Trampoline code page stuff. m_trampoline.S ---*/ 4/*--------------------------------------------------------------------*/ 5 6/* 7 This file is part of Valgrind, a dynamic binary instrumentation 8 framework. 9 10 Copyright (C) 2000-2015 Julian Seward 11 jseward@acm.org 12 Copyright (C) 2006-2015 OpenWorks LLP 13 info@open-works.co.uk 14 15 This program is free software; you can redistribute it and/or 16 modify it under the terms of the GNU General Public License as 17 published by the Free Software Foundation; either version 2 of the 18 License, or (at your option) any later version. 19 20 This program is distributed in the hope that it will be useful, but 21 WITHOUT ANY WARRANTY; without even the implied warranty of 22 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 23 General Public License for more details. 24 25 You should have received a copy of the GNU General Public License 26 along with this program; if not, write to the Free Software 27 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 28 02111-1307, USA. 29 30 The GNU General Public License is contained in the file COPYING. 31*/ 32 33#include "pub_core_basics_asm.h" 34#include "pub_core_vkiscnums_asm.h" 35 36/* ------------------ SIMULATED CPU HELPERS ------------------ */ 37/* 38 Replacements for some functions to do with vsyscalls and signals. 39 This code runs on the simulated CPU. 40*/ 41 42/*---------------------- x86-linux ----------------------*/ 43#if defined(VGP_x86_linux) 44 45# define UD2_16 ud2 ; ud2 ; ud2 ; ud2 ;ud2 ; ud2 ; ud2 ; ud2 46# define UD2_64 UD2_16 ; UD2_16 ; UD2_16 ; UD2_16 47# define UD2_256 UD2_64 ; UD2_64 ; UD2_64 ; UD2_64 48# define UD2_1024 UD2_256 ; UD2_256 ; UD2_256 ; UD2_256 49# define UD2_PAGE UD2_1024 ; UD2_1024 ; UD2_1024 ; UD2_1024 50 51 /* a leading page of unexecutable code */ 52 UD2_PAGE 53 54.global VG_(trampoline_stuff_start) 55VG_(trampoline_stuff_start): 56 57.global VG_(x86_linux_SUBST_FOR_sigreturn) 58VG_(x86_linux_SUBST_FOR_sigreturn): 59 /* This is a very specific sequence which GDB uses to 60 recognize signal handler frames. Also gcc: see 61 x86_fallback_frame_state() in 62 gcc-4.1.0/gcc/config/i386/linux-unwind.h */ 63 popl %eax 64 movl $ __NR_sigreturn, %eax 65 int $0x80 66 ud2 67 68.global VG_(x86_linux_SUBST_FOR_rt_sigreturn) 69VG_(x86_linux_SUBST_FOR_rt_sigreturn): 70 /* Likewise for rt signal frames */ 71 movl $ __NR_rt_sigreturn, %eax 72 int $0x80 73 ud2 74 75/* There's no particular reason that this needs to be handwritten 76 assembly, but since that's what this file contains, here's a 77 simple index implementation (written in C and compiled by gcc.) 78 79 unsigned char* REDIR_FOR_index ( const char* s, int c ) 80 { 81 unsigned char ch = (unsigned char)((unsigned int)c); 82 unsigned char* p = (unsigned char*)s; 83 while (1) { 84 if (*p == ch) return p; 85 if (*p == 0) return 0; 86 p++; 87 } 88 } 89*/ 90.global VG_(x86_linux_REDIR_FOR_index) 91.type VG_(x86_linux_REDIR_FOR_index), @function 92VG_(x86_linux_REDIR_FOR_index): 93 pushl %ebp 94 movl %esp, %ebp 95 movl 8(%ebp), %eax 96 movzbl 12(%ebp), %ecx 97 movzbl (%eax), %edx 98 cmpb %dl, %cl 99 jne .L9 100 jmp .L2 101.L11: 102 addl $1, %eax 103 movzbl (%eax), %edx 104 cmpb %dl, %cl 105 je .L2 106.L9: 107 testb %dl, %dl 108 jne .L11 109 xorl %eax, %eax 110.L2: 111 popl %ebp 112 ret 113.size VG_(x86_linux_REDIR_FOR_index), .-VG_(x86_linux_REDIR_FOR_index) 114 115/* There's no particular reason that this needs to be handwritten 116 assembly, but since that's what this file contains, here's a 117 simple strlen implementation (written in C and compiled by gcc.) 118*/ 119.global VG_(x86_linux_REDIR_FOR_strlen) 120.type VG_(x86_linux_REDIR_FOR_strlen), @function 121VG_(x86_linux_REDIR_FOR_strlen): 122 pushl %ebp 123 movl %esp, %ebp 124 movl 8(%ebp), %edx 125 movl %edx, %eax 126 jmp 2f 1271: incl %eax 1282: cmpb $0, (%eax) 129 jne 1b 130 subl %edx, %eax 131 popl %ebp 132 ret 133.size VG_(x86_linux_REDIR_FOR_strlen), .-VG_(x86_linux_REDIR_FOR_strlen) 134 135 136.global VG_(trampoline_stuff_end) 137VG_(trampoline_stuff_end): 138 139 /* and a trailing page of unexecutable code */ 140 UD2_PAGE 141 142# undef UD2_16 143# undef UD2_64 144# undef UD2_256 145# undef UD2_1024 146# undef UD2_PAGE 147 148/*---------------------- amd64-linux ----------------------*/ 149#else 150#if defined(VGP_amd64_linux) 151 152# define UD2_16 ud2 ; ud2 ; ud2 ; ud2 ;ud2 ; ud2 ; ud2 ; ud2 153# define UD2_64 UD2_16 ; UD2_16 ; UD2_16 ; UD2_16 154# define UD2_256 UD2_64 ; UD2_64 ; UD2_64 ; UD2_64 155# define UD2_1024 UD2_256 ; UD2_256 ; UD2_256 ; UD2_256 156# define UD2_PAGE UD2_1024 ; UD2_1024 ; UD2_1024 ; UD2_1024 157 158 /* a leading page of unexecutable code */ 159 UD2_PAGE 160 161.global VG_(trampoline_stuff_start) 162VG_(trampoline_stuff_start): 163 164.global VG_(amd64_linux_SUBST_FOR_rt_sigreturn) 165VG_(amd64_linux_SUBST_FOR_rt_sigreturn): 166 /* This is a very specific sequence which GDB uses to 167 recognize signal handler frames. */ 168 movq $__NR_rt_sigreturn, %rax 169 syscall 170 ud2 171 172.global VG_(amd64_linux_REDIR_FOR_vgettimeofday) 173.type VG_(amd64_linux_REDIR_FOR_vgettimeofday), @function 174VG_(amd64_linux_REDIR_FOR_vgettimeofday): 175.LfnB2: 176 movq $__NR_gettimeofday, %rax 177 syscall 178 ret 179.LfnE2: 180.size VG_(amd64_linux_REDIR_FOR_vgettimeofday), .-.LfnB2 181 182.global VG_(amd64_linux_REDIR_FOR_vtime) 183.type VG_(amd64_linux_REDIR_FOR_vtime), @function 184VG_(amd64_linux_REDIR_FOR_vtime): 185.LfnB3: 186 movq $__NR_time, %rax 187 syscall 188 ret 189.LfnE3: 190.size VG_(amd64_linux_REDIR_FOR_vtime), .-.LfnB3 191 192.global VG_(amd64_linux_REDIR_FOR_vgetcpu) 193.type VG_(amd64_linux_REDIR_FOR_vgetcpu), @function 194VG_(amd64_linux_REDIR_FOR_vgetcpu): 195.LfnB4: 196 movq $__NR_getcpu, %rax 197 syscall 198 ret 199.LfnE4: 200.size VG_(amd64_linux_REDIR_FOR_vgetcpu), .-.LfnB4 201 202/* There's no particular reason that this needs to be handwritten 203 assembly, but since that's what this file contains, here's a 204 simple strlen implementation (written in C and compiled by gcc.) 205*/ 206.global VG_(amd64_linux_REDIR_FOR_strlen) 207.type VG_(amd64_linux_REDIR_FOR_strlen), @function 208VG_(amd64_linux_REDIR_FOR_strlen): 209.LfnB5: 210 xorl %eax, %eax 211 cmpb $0, (%rdi) 212 movq %rdi, %rdx 213 je .L41 214.L40: addq $1, %rdx 215 cmpb $0, (%rdx) 216 jne .L40 217 movq %rdx, %rax 218 subq %rdi, %rax 219.L41: ret 220.LfnE5: 221.size VG_(amd64_linux_REDIR_FOR_strlen), .-VG_(amd64_linux_REDIR_FOR_strlen) 222 223 224/* A CIE for the above four functions, followed by their FDEs */ 225 .section .eh_frame,"a",@progbits 226.Lframe1: 227 .long .LEcie1-.LScie1 228.LScie1: 229 .long 0x0 230 .byte 0x1 231 .string "zR" 232 .uleb128 0x1 233 .sleb128 -8 234 .byte 0x10 235 .uleb128 0x1 236 .byte 0x3 237 .byte 0xc 238 .uleb128 0x7 239 .uleb128 0x8 240 .byte 0x90 241 .uleb128 0x1 242 .align 8 243.LEcie1: 244.LSfde2: 245 .long .LEfde2-.LASfde2 246.LASfde2: 247 .long .LASfde2-.Lframe1 248 .long .LfnB2 249 .long .LfnE2-.LfnB2 250 .uleb128 0x0 251 .align 8 252.LEfde2: 253.LSfde3: 254 .long .LEfde3-.LASfde3 255.LASfde3: 256 .long .LASfde3-.Lframe1 257 .long .LfnB3 258 .long .LfnE3-.LfnB3 259 .uleb128 0x0 260 .align 8 261.LEfde3: 262.LSfde4: 263 .long .LEfde4-.LASfde4 264.LASfde4: 265 .long .LASfde4-.Lframe1 266 .long .LfnB4 267 .long .LfnE4-.LfnB4 268 .uleb128 0x0 269 .align 8 270.LEfde4: 271.LSfde5: 272 .long .LEfde5-.LASfde5 273.LASfde5: 274 .long .LASfde5-.Lframe1 275 .long .LfnB5 276 .long .LfnE5-.LfnB5 277 .uleb128 0x0 278 .align 8 279.LEfde5: 280 .previous 281 282.global VG_(trampoline_stuff_end) 283VG_(trampoline_stuff_end): 284 285 /* and a trailing page of unexecutable code */ 286 UD2_PAGE 287 288# undef UD2_16 289# undef UD2_64 290# undef UD2_256 291# undef UD2_1024 292# undef UD2_PAGE 293 294/*---------------- ppc32-linux ----------------*/ 295#else 296#if defined(VGP_ppc32_linux) 297 298# define UD2_16 trap ; trap ; trap; trap 299# define UD2_64 UD2_16 ; UD2_16 ; UD2_16 ; UD2_16 300# define UD2_256 UD2_64 ; UD2_64 ; UD2_64 ; UD2_64 301# define UD2_1024 UD2_256 ; UD2_256 ; UD2_256 ; UD2_256 302# define UD2_PAGE UD2_1024 ; UD2_1024 ; UD2_1024 ; UD2_1024 303 304 /* a leading page of unexecutable code */ 305 UD2_PAGE 306 307.global VG_(trampoline_stuff_start) 308VG_(trampoline_stuff_start): 309 310.global VG_(ppc32_linux_SUBST_FOR_sigreturn) 311VG_(ppc32_linux_SUBST_FOR_sigreturn): 312 li 0,__NR_sigreturn 313 sc 314 .long 0 /*illegal insn*/ 315 316.global VG_(ppc32_linux_SUBST_FOR_rt_sigreturn) 317VG_(ppc32_linux_SUBST_FOR_rt_sigreturn): 318 li 0,__NR_rt_sigreturn 319 sc 320 .long 0 /*illegal insn*/ 321 322/* There's no particular reason that this needs to be handwritten 323 assembly, but since that's what this file contains, here's a 324 simple strlen implementation (written in C and compiled by gcc.) 325*/ 326.global VG_(ppc32_linux_REDIR_FOR_strlen) 327.type VG_(ppc32_linux_REDIR_FOR_strlen), @function 328VG_(ppc32_linux_REDIR_FOR_strlen): 329 lbz 4,0(3) 330 li 9,0 331 cmpwi 0,4,0 332 beq- 0,.L18 333.L19: 334 lbzu 5,1(3) 335 addi 9,9,1 336 cmpwi 0,5,0 337 bne+ 0,.L19 338.L18: 339 mr 3,9 340 blr 341.size VG_(ppc32_linux_REDIR_FOR_strlen), .-VG_(ppc32_linux_REDIR_FOR_strlen) 342 343/* Ditto strcmp */ 344.global VG_(ppc32_linux_REDIR_FOR_strcmp) 345.type VG_(ppc32_linux_REDIR_FOR_strcmp), @function 346VG_(ppc32_linux_REDIR_FOR_strcmp): 347.L20: 348 lbz 0,0(3) 349 cmpwi 7,0,0 350 bne- 7,.L21 351 lbz 0,0(4) 352 li 11,0 353 cmpwi 7,0,0 354 beq- 7,.L22 355.L21: 356 lbz 0,0(3) 357 li 11,-1 358 cmpwi 7,0,0 359 beq- 7,.L22 360 lbz 0,0(4) 361 li 11,1 362 cmpwi 7,0,0 363 beq- 7,.L22 364 lbz 9,0(3) 365 lbz 0,0(4) 366 li 11,-1 367 cmplw 7,9,0 368 blt- 7,.L22 369 lbz 9,0(3) 370 lbz 0,0(4) 371 li 11,1 372 addi 3,3,1 373 addi 4,4,1 374 cmplw 7,9,0 375 ble+ 7,.L20 376.L22: 377 mr 3,11 378 blr 379.size VG_(ppc32_linux_REDIR_FOR_strcmp), .-VG_(ppc32_linux_REDIR_FOR_strcmp) 380 381/* Ditto index/strchr */ 382.global VG_(ppc32_linux_REDIR_FOR_strchr) 383.type VG_(ppc32_linux_REDIR_FOR_strchr), @function 384VG_(ppc32_linux_REDIR_FOR_strchr): 385 lbz 0,0(3) 386 rlwinm 4,4,0,0xff 387 cmpw 7,4,0 388 beqlr 7 389 cmpwi 7,0,0 390 bne 7,.L308 391 b .L304 392.L309: 393 beq 6,.L304 394.L308: 395 lbzu 0,1(3) 396 cmpw 7,4,0 397 cmpwi 6,0,0 398 bne 7,.L309 399 blr 400.L304: 401 li 3,0 402 blr 403.size VG_(ppc32_linux_REDIR_FOR_strchr),.-VG_(ppc32_linux_REDIR_FOR_strchr) 404 405.global VG_(trampoline_stuff_end) 406VG_(trampoline_stuff_end): 407 408 /* and a trailing page of unexecutable code */ 409 UD2_PAGE 410 411# undef UD2_16 412# undef UD2_64 413# undef UD2_256 414# undef UD2_1024 415# undef UD2_PAGE 416 417/*---------------- ppc64-linux ----------------*/ 418#else 419#if defined(VGP_ppc64be_linux) || defined(VGP_ppc64le_linux) 420 421# define UD2_16 trap ; trap ; trap; trap 422# define UD2_64 UD2_16 ; UD2_16 ; UD2_16 ; UD2_16 423# define UD2_256 UD2_64 ; UD2_64 ; UD2_64 ; UD2_64 424# define UD2_1024 UD2_256 ; UD2_256 ; UD2_256 ; UD2_256 425# define UD2_PAGE UD2_1024 ; UD2_1024 ; UD2_1024 ; UD2_1024 426 427 /* a leading page of unexecutable code */ 428 UD2_PAGE 429 430.global VG_(trampoline_stuff_start) 431VG_(trampoline_stuff_start): 432 433.global VG_(ppc64_linux_SUBST_FOR_rt_sigreturn) 434VG_(ppc64_linux_SUBST_FOR_rt_sigreturn): 435 li 0,__NR_rt_sigreturn 436 sc 437 .long 0 /*illegal insn*/ 438 439 /* See comment in pub_core_trampoline.h for what this is for */ 440.global VG_(ppctoc_magic_redirect_return_stub) 441VG_(ppctoc_magic_redirect_return_stub): 442 trap 443 444 /* this function is written using the "dotless" ABI convention */ 445 .align 2 446 .globl VG_(ppc64_linux_REDIR_FOR_strlen) 447#if !defined VGP_ppc64be_linux || _CALL_ELF == 2 448 /* Little Endian uses ELF version 2 */ 449 .type VG_(ppc64_linux_REDIR_FOR_strlen),@function 450VG_(ppc64_linux_REDIR_FOR_strlen): 451#else 452 /* Big Endian uses ELF version 1 */ 453 .section ".opd","aw" 454 .align 3 455VG_(ppc64_linux_REDIR_FOR_strlen): 456 .quad .L.VG_(ppc64_linux_REDIR_FOR_strlen),.TOC.@tocbase,0 457 .previous 458 .size VG_(ppc64_linux_REDIR_FOR_strlen), \ 459 .L0end-.L.VG_(ppc64_linux_REDIR_FOR_strlen) 460 .type VG_(ppc64_linux_REDIR_FOR_strlen), @function 461 462.L.VG_(ppc64_linux_REDIR_FOR_strlen): 463#endif 464#if _CALL_ELF == 2 4650: addis 2,12,.TOC.-0b@ha 466 addi 2,2,.TOC.-0b@l 467 .localentry VG_(ppc64_linux_REDIR_FOR_strlen), .-VG_(ppc64_linux_REDIR_FOR_strlen) 468#endif 469 mr 9,3 470 lbz 0,0(3) 471 li 3,0 472 cmpwi 7,0,0 473 beqlr 7 474 li 3,0 475.L01: 476 addi 0,3,1 477 extsw 3,0 478 lbzx 0,9,3 479 cmpwi 7,0,0 480 bne 7,.L01 481 blr 482 483#if !defined VGP_ppc64be_linux || _CALL_ELF == 2 484 .size VG_(ppc64_linux_REDIR_FOR_strlen),.-VG_(ppc64_linux_REDIR_FOR_strlen) 485#else 486 .size VG_(ppc64_linux_REDIR_FOR_strlen),.-.L.VG_(ppc64_linux_REDIR_FOR_strlen) 487#endif 488 .long 0 489 .byte 0,0,0,0,0,0,0,0 490.L0end: 491 492 /* this function is written using the "dotless" ABI convention */ 493 .align 2 494 .globl VG_(ppc64_linux_REDIR_FOR_strchr) 495#if !defined VGP_ppc64be_linux || _CALL_ELF == 2 496 .type VG_(ppc64_linux_REDIR_FOR_strchr),@function 497VG_(ppc64_linux_REDIR_FOR_strchr): 498#else 499 .section ".opd","aw" 500 .align 3 501VG_(ppc64_linux_REDIR_FOR_strchr): 502 .quad .L.VG_(ppc64_linux_REDIR_FOR_strchr),.TOC.@tocbase,0 503 .previous 504 .size VG_(ppc64_linux_REDIR_FOR_strchr), \ 505 .L1end-.L.VG_(ppc64_linux_REDIR_FOR_strchr) 506 .type VG_(ppc64_linux_REDIR_FOR_strchr),@function 507 508.L.VG_(ppc64_linux_REDIR_FOR_strchr): 509#endif 510#if _CALL_ELF == 2 5110: addis 2,12,.TOC.-0b@ha 512 addi 2,2,.TOC.-0b@l 513 .localentry VG_(ppc64_linux_REDIR_FOR_strchr), .-VG_(ppc64_linux_REDIR_FOR_strchr) 514#endif 515 lbz 0,0(3) 516 rldicl 4,4,0,56 517 cmpw 7,4,0 518 beqlr 7 519 cmpdi 7,0,0 520 bne 7,.L18 521 b .L14 522#if !defined VGP_ppc64be_linux || _CALL_ELF == 2 523 .size VG_(ppc64_linux_REDIR_FOR_strchr),.-VG_(ppc64_linux_REDIR_FOR_strchr) 524#else 525 .size VG_(ppc64_linux_REDIR_FOR_strchr),.-.L.VG_(ppc64_linux_REDIR_FOR_strchr) 526#endif 527.L19: 528 beq 6,.L14 529.L18: 530 lbzu 0,1(3) 531 cmpw 7,4,0 532 cmpdi 6,0,0 533 bne 7,.L19 534 blr 535.L14: 536 li 3,0 537 blr 538 .long 0 539 .byte 0,0,0,0,0,0,0,0 540.L1end: 541 542 543.global VG_(trampoline_stuff_end) 544VG_(trampoline_stuff_end): 545 546 /* and a trailing page of unexecutable code */ 547 UD2_PAGE 548 549# undef UD2_16 550# undef UD2_64 551# undef UD2_256 552# undef UD2_1024 553# undef UD2_PAGE 554 555/*---------------- arm-linux ----------------*/ 556#else 557#if defined(VGP_arm_linux) 558 559# define UD2_4 .word 0xFFFFFFFF 560# define UD2_16 UD2_4 ; UD2_4 ; UD2_4 ; UD2_4 561# define UD2_64 UD2_16 ; UD2_16 ; UD2_16 ; UD2_16 562# define UD2_256 UD2_64 ; UD2_64 ; UD2_64 ; UD2_64 563# define UD2_1024 UD2_256 ; UD2_256 ; UD2_256 ; UD2_256 564# define UD2_PAGE UD2_1024 ; UD2_1024 ; UD2_1024 ; UD2_1024 565 566 /* a leading page of unexecutable code */ 567 UD2_PAGE 568 569.global VG_(trampoline_stuff_start) 570VG_(trampoline_stuff_start): 571 572.global VG_(arm_linux_SUBST_FOR_sigreturn) 573.type VG_(arm_linux_SUBST_FOR_sigreturn),#function 574VG_(arm_linux_SUBST_FOR_sigreturn): 575 mov r7, # __NR_sigreturn 576 svc #0 577 .long 0xFFFFFFFF /*illegal insn*/ 578.size VG_(arm_linux_SUBST_FOR_sigreturn), .-VG_(arm_linux_SUBST_FOR_sigreturn) 579 580.global VG_(arm_linux_SUBST_FOR_rt_sigreturn) 581.type VG_(arm_linux_SUBST_FOR_rt_sigreturn),#function 582VG_(arm_linux_SUBST_FOR_rt_sigreturn): 583 mov r7, # __NR_rt_sigreturn 584 svc #0 585 .long 0xFFFFFFFF /*illegal insn*/ 586.size VG_(arm_linux_SUBST_FOR_rt_sigreturn), .-VG_(arm_linux_SUBST_FOR_rt_sigreturn) 587 588.global VG_(arm_linux_REDIR_FOR_strlen) 589VG_(arm_linux_REDIR_FOR_strlen): 590 mov r2, r0 591 ldrb r0, [r0, #0] @ zero_extendqisi2 592 @ lr needed for prologue 593 cmp r0, #0 594 bxeq lr 595 mov r0, #0 596.L5: 597 add r0, r0, #1 598 ldrb r3, [r0, r2] @ zero_extendqisi2 599 cmp r3, #0 600 bne .L5 601 bx lr 602 UD2_4 603 604//.global VG_(arm_linux_REDIR_FOR_index) 605//VG_(arm_linux_REDIR_FOR_index): 606// ldrb r3, [r0, #0] @ zero_extendqisi2 607// and r1, r1, #255 608// cmp r3, r1 609// @ lr needed for prologue 610// bne .L9 611// bx lr 612//.L12: 613// ldrb r3, [r0, #1]! @ zero_extendqisi2 614// cmp r3, r1 615// beq .L11 616//.L9: 617// cmp r3, #0 618// bne .L12 619// mov r0, #0 620// bx lr 621//.L11: 622// bx lr 623// UD2_4 624 625.global VG_(arm_linux_REDIR_FOR_memcpy) 626VG_(arm_linux_REDIR_FOR_memcpy): 627 stmfd sp!, {r4, r5, lr} 628 subs lr, r2, #0 629 mov r5, r0 630 beq .L2 631 cmp r0, r1 632 bls .L4 633 add r3, r0, lr 634 add r1, lr, r1 635 cmp lr, #3 636 sub r4, r3, #1 637 sub r0, r1, #1 638 ble .L28 639 sub ip, r3, #5 640 sub r1, r1, #5 641.L8: 642 ldrb r3, [r1, #4] @ zero_extendqisi2 643 sub lr, lr, #4 644 strb r3, [ip, #4] 645 ldrb r2, [r1, #3] @ zero_extendqisi2 646 cmp lr, #3 647 strb r2, [ip, #3] 648 ldrb r3, [r1, #2] @ zero_extendqisi2 649 mov r4, ip 650 strb r3, [ip, #2] 651 ldrb r2, [r1, #1] @ zero_extendqisi2 652 mov r0, r1 653 strb r2, [ip, #1] 654 sub r1, r1, #4 655 sub ip, ip, #4 656 bgt .L8 657 cmp lr, #0 658 beq .L2 659.L28: 660 sub r2, lr, #1 661.L21: 662 sub r2, r2, #1 663 ldrb r3, [r0], #-1 @ zero_extendqisi2 664 cmn r2, #1 665 strb r3, [r4], #-1 666 bne .L21 667.L2: 668 mov r0, r5 669 ldmfd sp!, {r4, r5, pc} 670.L4: 671 bcs .L2 672 cmp lr, #3 673 mov ip, r0 674 ble .L29 675.L19: 676 ldrb r3, [r1, #0] @ zero_extendqisi2 677 sub lr, lr, #4 678 strb r3, [ip, #0] 679 ldrb r2, [r1, #1] @ zero_extendqisi2 680 cmp lr, #3 681 strb r2, [ip, #1] 682 ldrb r3, [r1, #2] @ zero_extendqisi2 683 strb r3, [ip, #2] 684 ldrb r2, [r1, #3] @ zero_extendqisi2 685 add r1, r1, #4 686 strb r2, [ip, #3] 687 add ip, ip, #4 688 bgt .L19 689 cmp lr, #0 690 beq .L2 691.L29: 692 sub r2, lr, #1 693.L20: 694 sub r2, r2, #1 695 ldrb r3, [r1], #1 @ zero_extendqisi2 696 cmn r2, #1 697 strb r3, [ip], #1 698 bne .L20 699 mov r0, r5 700 ldmfd sp!, {r4, r5, pc} 701 UD2_4 702 703.global VG_(arm_linux_REDIR_FOR_strcmp) 704VG_(arm_linux_REDIR_FOR_strcmp): 705.L64: 706 ldrb r3, [r0], #1 @ zero_extendqisi2 707 ldrb r2, [r1], #1 @ zero_extendqisi2 708 cmp r3, #0 709 beq .L67 710 cmp r3, r2 711 beq .L64 712 rsb r0, r2, r3 713 bx lr 714.L67: 715 rsb r0, r2, #0 716 bx lr 717 UD2_4 718 719.global VG_(trampoline_stuff_end) 720VG_(trampoline_stuff_end): 721 722 /* and a trailing page of unexecutable code */ 723 UD2_PAGE 724 725# undef UD2_4 726# undef UD2_16 727# undef UD2_64 728# undef UD2_256 729# undef UD2_1024 730# undef UD2_PAGE 731 732/*---------------- arm64-linux ----------------*/ 733#else 734#if defined(VGP_arm64_linux) 735 736# define UD2_4 .word 0xFFFFFFFF 737# define UD2_16 UD2_4 ; UD2_4 ; UD2_4 ; UD2_4 738# define UD2_64 UD2_16 ; UD2_16 ; UD2_16 ; UD2_16 739# define UD2_256 UD2_64 ; UD2_64 ; UD2_64 ; UD2_64 740# define UD2_1024 UD2_256 ; UD2_256 ; UD2_256 ; UD2_256 741# define UD2_PAGE UD2_1024 ; UD2_1024 ; UD2_1024 ; UD2_1024 742 743 /* a leading page of unexecutable code */ 744 UD2_PAGE 745 746.global VG_(trampoline_stuff_start) 747VG_(trampoline_stuff_start): 748 749.global VG_(arm64_linux_SUBST_FOR_rt_sigreturn) 750.type VG_(arm64_linux_SUBST_FOR_rt_sigreturn),#function 751VG_(arm64_linux_SUBST_FOR_rt_sigreturn): 752 mov x8, # __NR_rt_sigreturn 753 svc #0 754 .long 0xFFFFFFFF /*illegal insn*/ 755.size VG_(arm64_linux_SUBST_FOR_rt_sigreturn), \ 756 .-VG_(arm64_linux_SUBST_FOR_rt_sigreturn) 757 758.global VG_(arm64_linux_REDIR_FOR_strlen) 759.type VG_(arm64_linux_REDIR_FOR_strlen),#function 760VG_(arm64_linux_REDIR_FOR_strlen): 761 mov x2, x0 762 ldrb w0, [x0] 763 cbz w0, .L5 764 mov x0, 0 765.L4: 766 add x0, x0, 1 767 ldrb w1, [x2,x0] 768 cbnz w1, .L4 769 ret 770.L5: 771 mov x0, 0 772 ret 773.size VG_(arm64_linux_REDIR_FOR_strlen), .-VG_(arm64_linux_REDIR_FOR_strlen) 774 775.global VG_(arm64_linux_REDIR_FOR_index) 776.type VG_(arm64_linux_REDIR_FOR_index),#function 777VG_(arm64_linux_REDIR_FOR_index): 778 ldrb w2, [x0] 779 uxtb w1, w1 780 cmp w2, w1 781 beq .L11 782.L13: 783 cbz w2, .L16 784 ldrb w2, [x0,1]! 785 cmp w2, w1 786 bne .L13 787.L11: 788 ret 789.L16: 790 mov x0, 0 791 ret 792.size VG_(arm64_linux_REDIR_FOR_index), .-VG_(arm64_linux_REDIR_FOR_index) 793 794.global VG_(arm64_linux_REDIR_FOR_strcmp) 795.type VG_(arm64_linux_REDIR_FOR_strcmp),#function 796VG_(arm64_linux_REDIR_FOR_strcmp): 797 ldrb w2, [x0] 798 ldrb w3, [x1] 799 cmp w2, w3 800 bcc .L22 801.L21: 802 bhi .L25 803 cbz w2, .L26 804 ldrb w2, [x0,1]! 805 ldrb w3, [x1,1]! 806 cmp w2, w3 807 bcs .L21 808.L22: 809 mov x0, -1 810 ret 811.L25: 812 mov x0, 1 813 ret 814.L26: 815 mov x0, 0 816 ret 817.size VG_(arm64_linux_REDIR_FOR_strcmp), .-VG_(arm64_linux_REDIR_FOR_strcmp) 818 819.global VG_(trampoline_stuff_end) 820VG_(trampoline_stuff_end): 821 822 /* and a trailing page of unexecutable code */ 823 UD2_PAGE 824 825# undef UD2_4 826# undef UD2_16 827# undef UD2_64 828# undef UD2_256 829# undef UD2_1024 830# undef UD2_PAGE 831 832/*---------------- x86-darwin ----------------*/ 833#else 834#if defined(VGP_x86_darwin) 835 836 /* a leading page of unexecutable code */ 837.fill 2048, 2, 0x0b0f /* `ud2` */ 838 839.globl VG_(trampoline_stuff_start) 840VG_(trampoline_stuff_start): 841 842.globl VG_(x86_darwin_SUBST_FOR_sigreturn) 843VG_(x86_darwin_SUBST_FOR_sigreturn): 844 /* XXX does this need to have any special form? (cf x86-linux 845 version) */ 846 movl $ __NR_DARWIN_FAKE_SIGRETURN, %eax 847 int $0x80 848 ud2 849 850.globl VG_(x86_darwin_REDIR_FOR_strlen) 851VG_(x86_darwin_REDIR_FOR_strlen): 852 movl 4(%esp), %edx 853 movl %edx, %eax 854 jmp 1f 8550: 856 incl %eax 8571: 858 cmpb $0, (%eax) 859 jne 0b 860 subl %edx, %eax 861 ret 862 863.globl VG_(x86_darwin_REDIR_FOR_strcat) 864VG_(x86_darwin_REDIR_FOR_strcat): 865 pushl %esi 866 movl 8(%esp), %esi 867 movl 12(%esp), %ecx 868 movl %esi, %edx 869 jmp 1f 8700: 871 incl %edx 8721: 873 cmpb $0, (%edx) 874 jne 0b 8752: 876 movzbl (%ecx), %eax 877 incl %ecx 878 movb %al, (%edx) 879 incl %edx 880 testb %al, %al 881 jne 2b 882 movl %esi, %eax 883 popl %esi 884 ret 885 886 887.globl VG_(x86_darwin_REDIR_FOR_strcmp) 888VG_(x86_darwin_REDIR_FOR_strcmp): 889 movl 4(%esp), %edx 890 movl 8(%esp), %ecx 891 jmp 1f 8920: 893 incl %edx 894 incl %ecx 8951: 896 movzbl (%edx), %eax 897 testb %al, %al 898 je 2f 899 cmpb (%ecx), %al 900 je 0b 9012: 902 movzbl (%ecx),%edx 903 movzbl %al,%eax 904 subl %edx, %eax 905 ret 906 907 908.globl VG_(x86_darwin_REDIR_FOR_strcpy) 909VG_(x86_darwin_REDIR_FOR_strcpy): 910 pushl %ebp 911 movl %esp, %ebp 912 pushl %esi 913 movl 8(%ebp), %esi 914 movl 12(%ebp), %ecx 915 movl %esi, %edx 916 jmp 1f 9170: 918 incl %ecx 919 incl %edx 9201: 921 movzbl (%ecx), %eax 922 testb %al, %al 923 movb %al, (%edx) 924 jne 0b 925 movl %esi, %eax 926 popl %esi 927 leave 928 ret 929 930.globl VG_(x86_darwin_REDIR_FOR_strlcat) 931VG_(x86_darwin_REDIR_FOR_strlcat): 932 pushl %ebp 933 movl %esp, %ebp 934 pushl %edi 935 pushl %esi 936 subl $16, %esp 937 movl 8(%ebp), %esi 938 movl 16(%ebp), %ecx 939 movl %esi, %edx 940 leal (%ecx,%esi), %eax 941 jmp 1f 9420: 943 incl %edx 9441: 945 cmpl %edx, %eax 946 je 2f 947 cmpb $0, (%edx) 948 jne 0b 9492: 950 movl %edx, %edi 951 subl %esi, %edi 952 movl %ecx, %esi 953 subl %edi, %esi 954 je 3f 955 movl 12(%ebp), %eax 956 jmp 6f 9573: 958 movl 12(%ebp), %eax 959 movl %eax, (%esp) 960 call VG_(x86_darwin_REDIR_FOR_strlen) 961 jmp 7f 9624: 963 cmpl $1, %esi 964 je 5f 965 movb %cl, (%edx) 966 decl %esi 967 incl %edx 9685: 969 incl %eax 9706: 971 movzbl (%eax), %ecx 972 testb %cl, %cl 973 jne 4b 974 movb $0, (%edx) 975 subl 12(%ebp), %eax 9767: 977 addl $16, %esp 978 leal (%edi,%eax), %eax 979 popl %esi 980 popl %edi 981 leave 982 ret 983 984 985.globl VG_(trampoline_stuff_end) 986VG_(trampoline_stuff_end): 987 988 /* a trailing page of unexecutable code */ 989.fill 2048, 2, 0x0b0f /* `ud2` */ 990 991 992/*---------------- amd64-darwin ----------------*/ 993#else 994#if defined(VGP_amd64_darwin) 995 996 /* a leading page of unexecutable code */ 997.fill 2048, 2, 0x0b0f /* `ud2` */ 998 999.globl VG_(trampoline_stuff_start) 1000VG_(trampoline_stuff_start): 1001 1002.globl VG_(amd64_darwin_SUBST_FOR_sigreturn) 1003VG_(amd64_darwin_SUBST_FOR_sigreturn): 1004 /* XXX does this need to have any special form? (cf x86-linux 1005 version) */ 1006 movq $ __NR_DARWIN_FAKE_SIGRETURN, %rax 1007 syscall 1008 ud2 1009 1010.globl VG_(amd64_darwin_REDIR_FOR_strlen) 1011VG_(amd64_darwin_REDIR_FOR_strlen): 1012 movq %rdi, %rax 1013 jmp 1f 10140: 1015 incq %rax 10161: 1017 cmpb $0, (%rax) 1018 jne 0b 1019 subq %rdi, %rax 1020 ret 1021 1022.globl VG_(amd64_darwin_REDIR_FOR_strcat) 1023VG_(amd64_darwin_REDIR_FOR_strcat): 1024 movq %rdi, %rdx 1025 jmp 1f 10260: 1027 incq %rdx 10281: 1029 cmpb $0, (%rdx) 1030 jne 0b 10312: 1032 movzbl (%rsi), %eax 1033 incq %rsi 1034 movb %al, (%rdx) 1035 incq %rdx 1036 testb %al, %al 1037 jne 2b 1038 movq %rdi, %rax 1039 ret 1040 1041 1042.globl VG_(amd64_darwin_REDIR_FOR_strcmp) 1043VG_(amd64_darwin_REDIR_FOR_strcmp): 1044 jmp 1f 10450: 1046 incq %rdi 1047 incq %rsi 10481: 1049 movzbl (%rdi), %eax 1050 testb %al, %al 1051 je 2f 1052 cmpb (%rsi), %al 1053 je 0b 10542: 1055 movzbl (%rsi), %edx 1056 movzbl %al, %eax 1057 subl %edx, %eax 1058 ret 1059 1060.globl VG_(amd64_darwin_REDIR_FOR_strcpy) 1061VG_(amd64_darwin_REDIR_FOR_strcpy): 1062 pushq %rbp 1063 movq %rdi, %rdx 1064 movq %rsp, %rbp 1065 jmp 1f 10660: 1067 incq %rsi 1068 incq %rdx 10691: 1070 movzbl (%rsi), %eax 1071 testb %al, %al 1072 movb %al, (%rdx) 1073 jne 0b 1074 leave 1075 movq %rdi, %rax 1076 ret 1077 1078.globl VG_(amd64_darwin_REDIR_FOR_strlcat) 1079VG_(amd64_darwin_REDIR_FOR_strlcat): 1080 pushq %rbp 1081 leaq (%rdx,%rdi), %rax 1082 movq %rdi, %rcx 1083 movq %rsp, %rbp 1084 pushq %rbx 1085 subq $8, %rsp 1086 jmp 1f 10870: 1088 incq %rcx 10891: 1090 cmpq %rcx, %rax 1091 je 2f 1092 cmpb $0, (%rcx) 1093 jne 0b 10942: 1095 movq %rcx, %rbx 1096 subq %rdi, %rbx 1097 movq %rdx, %rdi 1098 subq %rbx, %rdi 1099 je 3f 1100 movq %rsi, %rax 1101 jmp 6f 11023: 1103 movq %rsi, %rdi 1104 call VG_(amd64_darwin_REDIR_FOR_strlen) 1105 jmp 7f 11064: 1107 cmpq $1, %rdi 1108 je 5f 1109 movb %dl, (%rcx) 1110 decq %rdi 1111 incq %rcx 11125: 1113 incq %rax 11146: 1115 movzbl (%rax), %edx 1116 testb %dl, %dl 1117 jne 4b 1118 movb $0, (%rcx) 1119 subq %rsi, %rax 11207: 1121 leaq (%rbx,%rax), %rax 1122 addq $8, %rsp 1123 popq %rbx 1124 leave 1125 ret 1126 1127.globl VG_(amd64_darwin_REDIR_FOR_arc4random) 1128VG_(amd64_darwin_REDIR_FOR_arc4random): 1129 /* not very random, hope dyld won't mind */ 1130 movq $0x76616c6772696e64, %rax 1131 ret 1132 1133.globl VG_(amd64_darwin_REDIR_FOR_strchr) 1134VG_(amd64_darwin_REDIR_FOR_strchr): 1135 pushq %rbp 1136 movq %rsp, %rbp 1137 movb (%rdi), %cl 1138 cmpb %sil, %cl 1139 jne 1f 1140 movq %rdi, %rax 1141 popq %rbp 1142 ret 11431: 1144 testb %cl, %cl 1145 movl $0, %eax 1146 je 2f 1147 movb 1(%rdi), %cl 1148 incq %rdi 1149 cmpb %sil, %cl 1150 movq %rdi, %rax 1151 jne 1b 11522: 1153 popq %rbp 1154 ret 1155 1156.globl VG_(trampoline_stuff_end) 1157VG_(trampoline_stuff_end): 1158 1159 /* a trailing page of unexecutable code */ 1160.fill 2048, 2, 0x0b0f /* `ud2` */ 1161 1162 1163/*---------------- s390x-linux ----------------*/ 1164#else 1165#if defined(VGP_s390x_linux) 1166 1167 /* a leading page of unexecutable code */ 1168 .fill 2048, 2, 0x0000 1169 1170.global VG_(trampoline_stuff_start) 1171VG_(trampoline_stuff_start): 1172 1173.global VG_(s390x_linux_SUBST_FOR_sigreturn) 1174VG_(s390x_linux_SUBST_FOR_sigreturn): 1175 svc __NR_sigreturn 1176 .short 0 1177 1178.global VG_(s390x_linux_SUBST_FOR_rt_sigreturn) 1179VG_(s390x_linux_SUBST_FOR_rt_sigreturn): 1180 /* Old gcc unwinding code checks for a sig(_rt)_return svc and then 1181 for ra = cfa to decide if it is a sig_rt_frame or not. Since we 1182 set ra to this trampoline, but the cfa is still in the stack, 1183 the unwinder thinks, that this is a non-rt frame and causes a 1184 crash in the gcc unwinder - which is used by the thread library 1185 and others. Therefore we add a lr 1,1 nop, to let the gcc 1186 unwinder bail out gracefully. This might also affect unwinding 1187 across the signal frame - tough luck. fixs390 */ 1188 lr 1,1 1189 svc __NR_rt_sigreturn 1190 .short 0 1191 1192.global VG_(s390x_linux_REDIR_FOR_index) 1193.type VG_(s390x_linux_REDIR_FOR_index),@function 1194VG_(s390x_linux_REDIR_FOR_index): 1195# 1196# %r2 = addess of string 1197# %r3 = character to find 1198# 1199 lghi %r0,255 1200 ngr %r0,%r3 # r0 = (unsigned char)r3 1201 lghi %r4,0 1202.L1: 1203 llgc %r1,0(%r2) # r1 = byte from string 1204 cr %r1,%r0 # compare 1205 ber %r14 # return if found 1206 cr %r1,%r4 # end of string ? 1207 je .L2 1208 aghi %r2,1 # increment r2 1209 j .L1 1210.L2: lghi %r2,0 # return value 0 1211 br %r14 1212.size VG_(s390x_linux_REDIR_FOR_index), .-VG_(s390x_linux_REDIR_FOR_index) 1213 1214.globl VG_(trampoline_stuff_end) 1215VG_(trampoline_stuff_end): 1216 .fill 2048, 2, 0x0000 1217 1218/*---------------------- mips32-linux ----------------------*/ 1219#else 1220#if defined(VGP_mips32_linux) 1221 1222# define UD2_16 trap ; trap ; trap; trap 1223# define UD2_64 UD2_16 ; UD2_16 ; UD2_16 ; UD2_16 1224# define UD2_256 UD2_64 ; UD2_64 ; UD2_64 ; UD2_64 1225# define UD2_1024 UD2_256 ; UD2_256 ; UD2_256 ; UD2_256 1226# define UD2_PAGE UD2_1024 ; UD2_1024 ; UD2_1024 ; UD2_1024 1227 1228 1229.global VG_(trampoline_stuff_start) 1230VG_(trampoline_stuff_start): 1231 1232.global VG_(mips32_linux_SUBST_FOR_sigreturn) 1233VG_(mips32_linux_SUBST_FOR_sigreturn): 1234 li $v0,__NR_sigreturn 1235 syscall 1236 nop 1237 .long 0 /*illegal insn*/ 1238 1239.global VG_(mips32_linux_SUBST_FOR_rt_sigreturn) 1240VG_(mips32_linux_SUBST_FOR_rt_sigreturn): 1241 li $v0,__NR_rt_sigreturn 1242 syscall 1243 nop 1244 .long 0 /*illegal insn*/ 1245 1246/* There's no particular reason that this needs to be handwritten 1247 assembly, but since that's what this file contains, here's a 1248 simple strlen implementation (written in C and compiled by gcc.) 1249*/ 1250.global VG_(mips32_linux_REDIR_FOR_strlen) 1251.type VG_(mips32_linux_REDIR_FOR_strlen), @function 1252VG_(mips32_linux_REDIR_FOR_strlen): 1253 li $v0, 0 1254 //la $a0, string 1255 j strlen_cond 1256 strlen_loop: 1257 addi $v0, $v0, 1 1258 addi $a0, $a0, 1 1259 strlen_cond: 1260 lbu $t0, ($a0) 1261 bne $t0, $zero, strlen_loop 1262 jr $ra 1263 1264.size VG_(mips32_linux_REDIR_FOR_strlen), .-VG_(mips32_linux_REDIR_FOR_strlen) 1265 1266.global VG_(trampoline_stuff_end) 1267VG_(trampoline_stuff_end): 1268 1269 1270# undef UD2_16 1271# undef UD2_64 1272# undef UD2_256 1273# undef UD2_1024 1274# undef UD2_PAGE 1275 1276/*---------------------- mips64-linux ----------------------*/ 1277#else 1278#if defined(VGP_mips64_linux) 1279 1280# define UD2_16 trap ; trap ; trap; trap 1281# define UD2_64 UD2_16 ; UD2_16 ; UD2_16 ; UD2_16 1282# define UD2_256 UD2_64 ; UD2_64 ; UD2_64 ; UD2_64 1283# define UD2_1024 UD2_256 ; UD2_256 ; UD2_256 ; UD2_256 1284# define UD2_PAGE UD2_1024 ; UD2_1024 ; UD2_1024 ; UD2_1024 1285 1286.global VG_(trampoline_stuff_start) 1287VG_(trampoline_stuff_start): 1288 1289.global VG_(mips64_linux_SUBST_FOR_rt_sigreturn) 1290VG_(mips64_linux_SUBST_FOR_rt_sigreturn): 1291 li $2,__NR_rt_sigreturn 1292 syscall 1293 nop 1294 .long 0 /*illegal insn*/ 1295 1296/* There's no particular reason that this needs to be handwritten 1297 assembly, but since that's what this file contains, here's a 1298 simple strlen implementation (written in C and compiled by gcc.) 1299*/ 1300.global VG_(mips64_linux_REDIR_FOR_strlen) 1301.type VG_(mips64_linux_REDIR_FOR_strlen), @function 1302VG_(mips64_linux_REDIR_FOR_strlen): 1303 lbu $12, 0($4) 1304 li $13, 0 1305 beq $12, $0, M01 1306 nop 1307 1308M02: 1309 addiu $13, $13, 1 1310 addiu $4, $4, 1 1311 lbu $12, 0($4) 1312 bne $12, $0, M02 1313 nop 1314 1315M01: 1316 move $2, $13 1317 jr $31 1318 nop 1319 1320.size VG_(mips64_linux_REDIR_FOR_strlen), .-VG_(mips64_linux_REDIR_FOR_strlen) 1321 1322.global VG_(trampoline_stuff_end) 1323VG_(trampoline_stuff_end): 1324 1325 1326# undef UD2_16 1327# undef UD2_64 1328# undef UD2_256 1329# undef UD2_1024 1330# undef UD2_PAGE 1331 1332/*---------------------- tilegx-linux ----------------------*/ 1333#else 1334#if defined(VGP_tilegx_linux) 1335 1336# define UD2_16 ill ; ill 1337# define UD2_64 UD2_16 ; UD2_16 ; UD2_16 ; UD2_16 1338# define UD2_256 UD2_64 ; UD2_64 ; UD2_64 ; UD2_64 1339# define UD2_1024 UD2_256 ; UD2_256 ; UD2_256 ; UD2_256 1340# define UD2_4K UD2_1024 ; UD2_1024 ; UD2_1024 ; UD2_1024 1341# define UD2_16K UD2_4K ; UD2_4K ; UD2_4K ; UD2_4K 1342# define UD2_PAGE UD2_16K ; UD2_16K ; UD2_16K ; UD2_16K 1343 /* a leading page of unexecutable code */ 1344 UD2_PAGE 1345 1346.global VG_(trampoline_stuff_start) 1347VG_(trampoline_stuff_start): 1348 1349.global VG_(tilegx_linux_SUBST_FOR_rt_sigreturn) 1350VG_(tilegx_linux_SUBST_FOR_rt_sigreturn): 1351 /* This is a very specific sequence which GDB uses to 1352 recognize signal handler frames. */ 1353 moveli r10, __NR_rt_sigreturn 1354 swint1 1355 ill 1356 1357.global VG_(tilegx_linux_REDIR_FOR_vgettimeofday) 1358.type VG_(tilegx_linux_REDIR_FOR_vgettimeofday), @function 1359VG_(tilegx_linux_REDIR_FOR_vgettimeofday): 1360 moveli r10, __NR_gettimeofday 1361 swint1 1362 jrp lr 1363.size VG_(tilegx_linux_REDIR_FOR_vgettimeofday), .-VG_(tilegx_linux_REDIR_FOR_vgettimeofday) 1364 1365.global VG_(tilegx_linux_REDIR_FOR_vtime) 1366.type VG_(tilegx_linux_REDIR_FOR_vtime), @function 1367VG_(tilegx_linux_REDIR_FOR_vtime): 1368 moveli r10, __NR_gettimeofday 1369 swint1 1370 jrp lr 1371.size VG_(tilegx_linux_REDIR_FOR_vtime), .-VG_(tilegx_linux_REDIR_FOR_vtime) 1372 1373.global VG_(tilegx_linux_REDIR_FOR_strlen) 1374.type VG_(tilegx_linux_REDIR_FOR_strlen), @function 1375VG_(tilegx_linux_REDIR_FOR_strlen): 1376 { 1377 movei r1, 0 1378 beqz r0, 2f 1379 } 13801: { 1381 addi r1, r1, 1 1382 ld1s_add r2, r0, 1 1383 } 1384 bnezt r2, 1b 1385 addi r1, r1, -1 13862: move r0, r1 1387 jrp lr 1388.size VG_(tilegx_linux_REDIR_FOR_strlen), .-VG_(tilegx_linux_REDIR_FOR_strlen) 1389 1390.global VG_(trampoline_stuff_end) 1391VG_(trampoline_stuff_end): 1392 1393 /* and a trailing page of unexecutable code */ 1394 UD2_PAGE 1395 1396# undef UD2_16 1397# undef UD2_64 1398# undef UD2_256 1399# undef UD2_1024 1400# undef UD2_4K 1401# undef UD2_16K 1402# undef UD2_PAGE 1403 1404/*---------------- x86-solaris ----------------*/ 1405#else 1406#if defined(VGP_x86_solaris) 1407 1408.global VG_(trampoline_stuff_start) 1409VG_(trampoline_stuff_start): 1410 1411/* int strcmp(const char *s1, const char *s2); */ 1412.global VG_(x86_solaris_REDIR_FOR_strcmp) 1413.type VG_(x86_solaris_REDIR_FOR_strcmp), @function 1414VG_(x86_solaris_REDIR_FOR_strcmp): 1415 pushl %ebp /* establish a stack frame */ 1416 movl %esp, %ebp 1417 movl 8(%ebp), %edx /* get s1 */ 1418 movl 12(%esp), %ecx /* get s2 */ 1419 jmp 2f /* go compare the first characters */ 14201: 1421 incl %edx /* skip to the next s1 character */ 1422 incl %ecx /* skip to the next s2 character */ 14232: 1424 movzbl (%edx), %eax /* load a character from s1 */ 1425 testb %al, %al /* is it null? */ 1426 jz 3f /* yes, exit */ 1427 cmpb (%ecx), %al /* are the characters equal? */ 1428 je 1b /* yes, proceed with next characters */ 14293: 1430 movzbl (%ecx), %edx /* load a character from s2 */ 1431 subl %edx, %eax /* calculate the return value */ 1432 popl %ebp /* destroy the stack frame */ 1433 ret /* return to the caller */ 1434.size VG_(x86_solaris_REDIR_FOR_strcmp), .-VG_(x86_solaris_REDIR_FOR_strcmp) 1435 1436/* size_t strlen(const char *s); */ 1437.global VG_(x86_solaris_REDIR_FOR_strlen) 1438.type VG_(x86_solaris_REDIR_FOR_strlen), @function 1439VG_(x86_solaris_REDIR_FOR_strlen): 1440 pushl %ebp /* establish a stack frame */ 1441 movl %esp, %ebp 1442 movl 8(%ebp), %edx /* get s */ 1443 movl %edx, %eax /* copy s */ 1444 jmp 2f /* go handle the first character */ 14451: 1446 incl %eax /* skip to the next s character */ 14472: 1448 cmpb $0, (%eax) /* is the s character null? */ 1449 jne 1b /* no, go process the next character */ 1450 subl %edx, %eax /* calculate the return value */ 1451 popl %ebp /* destroy the stack frame */ 1452 ret /* return to the caller */ 1453.size VG_(x86_solaris_REDIR_FOR_strlen), .-VG_(x86_solaris_REDIR_FOR_strlen) 1454 1455.global VG_(trampoline_stuff_end) 1456VG_(trampoline_stuff_end): 1457 1458/*---------------- amd64-solaris ----------------*/ 1459#else 1460#if defined(VGP_amd64_solaris) 1461 1462.global VG_(trampoline_stuff_start) 1463VG_(trampoline_stuff_start): 1464 1465/* char *strcpy(char *restrict s1, const char *restrict s2); */ 1466.global VG_(amd64_solaris_REDIR_FOR_strcpy) 1467.type VG_(amd64_solaris_REDIR_FOR_strcpy), @function 1468VG_(amd64_solaris_REDIR_FOR_strcpy): 1469 pushq %rbp /* establish a stack frame */ 1470 movq %rsp, %rbp 1471 movq %rdi, %rdx /* copy s1 */ 14721: 1473 movzbl (%rsi), %eax /* load one input character */ 1474 movb %al, (%rdx) /* copy to output/s2 */ 1475 incq %rsi /* skip to the next output character */ 1476 incq %rdx /* skip to the next input character */ 1477 testb %al, %al /* is the copied character null? */ 1478 jnz 1b /* no, copy the next character */ 1479 leave /* destroy the stack frame */ 1480 movq %rdi, %rax /* set s1 as the return value */ 1481 ret /* return to the caller */ 1482.size VG_(amd64_solaris_REDIR_FOR_strcpy), .-VG_(amd64_solaris_REDIR_FOR_strcpy) 1483 1484/* char *strncpy(char *restrict s1, const char *restrict s2, size_t n); */ 1485.global VG_(amd64_solaris_REDIR_FOR_strncpy) 1486.type VG_(amd64_solaris_REDIR_FOR_strncpy), @function 1487VG_(amd64_solaris_REDIR_FOR_strncpy): 1488 pushq %rbp /* establish a stack frame */ 1489 movq %rsp, %rbp 1490 movq %rdi, %rcx /* copy s1 */ 14911: 1492 testq %rdx, %rdx /* is the remaining size zero? */ 1493 jz 3f /* yes, all done */ 1494 movzbl (%rsi), %eax /* load one input character */ 1495 movb %al, (%rcx) /* copy to output/s2 */ 1496 decq %rdx /* decrement the remaining size */ 1497 incq %rsi /* skip to the next output character */ 1498 incq %rcx /* skip to the next input character */ 1499 testb %al, %al /* is the copied character null? */ 1500 jnz 1b /* no, copy the next character */ 15012: 1502 testq %rdx, %rdx /* is the remaining size zero? */ 1503 jz 3f /* yes, all done */ 1504 movb $0, (%rdx) /* copy null to output/s2 */ 1505 decq %rdx /* decrement the remaining size */ 1506 incq %rsi /* skip to next output character */ 1507 jmp 2b /* proceed with the next character */ 15083: 1509 leave /* destroy the stack frame */ 1510 movq %rdi, %rax /* set s1 as the return value */ 1511 ret /* return to the caller */ 1512.size VG_(amd64_solaris_REDIR_FOR_strncpy), .-VG_(amd64_solaris_REDIR_FOR_strncpy) 1513 1514/* int strcmp(const char *s1, const char *s2); */ 1515.global VG_(amd64_solaris_REDIR_FOR_strcmp) 1516.type VG_(amd64_solaris_REDIR_FOR_strcmp), @function 1517VG_(amd64_solaris_REDIR_FOR_strcmp): 1518 pushq %rbp /* establish a stack frame */ 1519 movq %rsp, %rbp 1520 jmp 2f /* go compare the first characters */ 15211: 1522 incq %rdi /* skip to the next s1 character */ 1523 incq %rsi /* skip to the next s2 character */ 15242: 1525 movzbl (%rdi), %eax /* load a character from s1 */ 1526 testb %al, %al /* is it null? */ 1527 jz 3f /* yes, exit */ 1528 cmpb (%rsi), %al /* are the characters equal? */ 1529 je 1b /* yes, proceed with next characters */ 15303: 1531 movzbl (%rsi), %edx /* load a character from s2 */ 1532 subl %edx, %eax /* calculate the return value */ 1533 leave /* destroy the stack frame */ 1534 ret /* return to the caller */ 1535.size VG_(amd64_solaris_REDIR_FOR_strcmp), .-VG_(amd64_solaris_REDIR_FOR_strcmp) 1536 1537/* char *strcat(char *restrict s1, const char *restrict s2); */ 1538.global VG_(amd64_solaris_REDIR_FOR_strcat) 1539.type VG_(amd64_solaris_REDIR_FOR_strcat), @function 1540VG_(amd64_solaris_REDIR_FOR_strcat): 1541 pushq %rbp /* establish a stack frame */ 1542 movq %rsp, %rbp 1543 movq %rdi, %rdx /* copy s1 */ 1544 jmp 2f /* go handle the first character */ 15451: 1546 incq %rdx /* skip to the next s1 character */ 15472: 1548 cmpb $0, (%rdx) /* is the s1 character null? */ 1549 jne 1b /* no, go check the next character */ 15503: 1551 movzbl (%rsi), %eax /* load a character from s2 */ 1552 movb %al, (%rdx) /* copy the s2 character to s1 */ 1553 incq %rdx /* skip to the next s1 character */ 1554 incq %rsi /* skip to the next s2 character */ 1555 testb %al, %al /* was the character null? */ 1556 jnz 3b /* no, go copy the next character */ 1557 movq %rdi, %rax /* set s1 as the return value */ 1558 leave /* destroy the stack frame */ 1559 ret /* return to the caller */ 1560.size VG_(amd64_solaris_REDIR_FOR_strcat), .-VG_(amd64_solaris_REDIR_FOR_strcat) 1561 1562/* size_t strlen(const char *s); */ 1563.global VG_(amd64_solaris_REDIR_FOR_strlen) 1564.type VG_(amd64_solaris_REDIR_FOR_strlen), @function 1565VG_(amd64_solaris_REDIR_FOR_strlen): 1566 pushq %rbp /* establish a stack frame */ 1567 movq %rsp, %rbp 1568 movq %rdi, %rax /* copy s */ 1569 jmp 2f /* go handle the first character */ 15701: 1571 incq %rax /* skip to the next s character */ 15722: 1573 cmpb $0, (%rax) /* is the s character null? */ 1574 jne 1b /* no, go process the next character */ 1575 subq %rdi, %rax /* calculate the return value */ 1576 leave /* destroy the stack frame */ 1577 ret /* return to the caller */ 1578.size VG_(amd64_solaris_REDIR_FOR_strlen), .-VG_(amd64_solaris_REDIR_FOR_strlen) 1579 1580.global VG_(trampoline_stuff_end) 1581VG_(trampoline_stuff_end): 1582 1583/*---------------- unknown ----------------*/ 1584#else 1585# error Unknown platform 1586 1587#endif 1588#endif 1589#endif 1590#endif 1591#endif 1592#endif 1593#endif 1594#endif 1595#endif 1596#endif 1597#endif 1598#endif 1599#endif 1600#endif 1601 1602/* Let the linker know we don't need an executable stack */ 1603MARK_STACK_NO_EXEC 1604 1605/*--------------------------------------------------------------------*/ 1606/*--- end ---*/ 1607/*--------------------------------------------------------------------*/ 1608