1 2/*--------------------------------------------------------------------*/ 3/*--- Support for doing system calls. syscall-amd64-solaris.S ---*/ 4/*--------------------------------------------------------------------*/ 5 6/* 7 This file is part of Valgrind, a dynamic binary instrumentation 8 framework. 9 10 Copyright (C) 2014-2015 Petr Pavlu 11 setup@dagobah.cz 12 13 This program is free software; you can redistribute it and/or 14 modify it under the terms of the GNU General Public License as 15 published by the Free Software Foundation; either version 2 of the 16 License, or (at your option) any later version. 17 18 This program is distributed in the hope that it will be useful, but 19 WITHOUT ANY WARRANTY; without even the implied warranty of 20 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU 21 General Public License for more details. 22 23 You should have received a copy of the GNU General Public License 24 along with this program; if not, write to the Free Software 25 Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 26 02111-1307, USA. 27 28 The GNU General Public License is contained in the file COPYING. 29*/ 30 31#include "pub_core_basics_asm.h" 32 33#if defined(VGP_amd64_solaris) 34 35#include "pub_core_vkiscnums_asm.h" 36#include "libvex_guest_offsets.h" 37 38/* From vki-solaris.h, checked at startup by m_vki.c. */ 39#define VKI_SIG_SETMASK 3 40 41/* Prototype: 42 Int ML_(do_syscall_for_client_WRK)( 43 Int syscallno, // %rdi = %rbp-48 44 void *guest_state, // %rsi = %rbp-40 45 const vki_sigset_t *sysmask, // %rdx = %rbp-32 46 const vki_sigset_t *postmask, // %rcx = %rbp-24 47 UChar *cflag) // %r8 = %rbp-16 48*/ 49 50.macro ESTABLISH_STACKFRAME 51 /* Establish stack frame. */ 52 pushq %rbp 53 movq %rsp, %rbp 54 pushq %rbx /* save %rbx */ 55 56 /* We'll use %rbx instead of %rbp to address the stack frame after the 57 door syscall is finished because %rbp is cleared by the syscall. */ 58 movq %rsp, %rbx /* %rbx = %rbp - 8 */ 59 60 /* Push the parameters on the stack. */ 61 pushq %r8 /* store %r8 at %rbp-16 */ 62 pushq %rcx /* store %rcx at %rbp-24 */ 63 pushq %rdx /* store %rdx at %rbp-32 */ 64 pushq %rsi /* store %rsi at %rbp-40 */ 65 pushq %rdi /* store %rdi at %rbp-48 */ 66.endm 67 68.macro UNBLOCK_SIGNALS 69 /* Set the signal mask which should be current during the syscall. */ 70 /* Set up for sigprocmask(SIG_SETMASK, sysmask, postmask). */ 71 movq -24(%rbp), %rdx 72 movq -32(%rbp), %rsi 73 movq $VKI_SIG_SETMASK, %rdi 74 movq $__NR_sigprocmask, %rax 75 syscall 76 jc sigprocmask_failed /* sigprocmask failed */ 77.endm 78 79.macro REBLOCK_SIGNALS 80 /* Set up for sigprocmask(SIG_SETMASK, postmask, NULL). */ 81 movq $0, %rdx 82 movq -24(%rbp), %rsi 83 movq $VKI_SIG_SETMASK, %rdi 84 movq $__NR_sigprocmask, %rax 85 syscall 86 /* The syscall above changes the carry flag. This means that if the 87 syscall fails and we receive an interrupt after it then we've got 88 an invalid carry flag value in the fixup code. We don't care about 89 it because this syscall should never fail and if it does then we're 90 going to stop Valgrind anyway. */ 91 jc sigprocmask_failed /* sigprocmask failed */ 92.endm 93 94.macro SIMPLE_RETURN 95 xorq %rax, %rax /* SUCCESS */ 96 movq -8(%rbp), %rbx /* restore %rbx */ 97 movq %rbp, %rsp 98 popq %rbp 99 ret 100.endm 101 102sigprocmask_failed: 103 /* Failure: return 0x8000 | error code. */ 104 andq $0x7FFF, %rax 105 orq $0x8000, %rax 106 movq -8(%rbp), %rbx /* restore %rbx */ 107 movq %rbp, %rsp 108 popq %rbp 109 ret 110 111.globl ML_(do_syscall_for_client_WRK) 112ML_(do_syscall_for_client_WRK): 113 ESTABLISH_STACKFRAME 114 1151: /* Even though we can't take a signal until the sigprocmask completes, 116 start the range early. If %rip is in the range [1, 2), the syscall 117 hasn't been started yet. */ 118 UNBLOCK_SIGNALS 119 120 /* Copy syscall parameters. */ 121 /* do_syscall8 */ 122 /* 6 register parameters. */ 123 movq -40(%rbp), %rax 124 movq OFFSET_amd64_RDI(%rax), %rdi 125 movq OFFSET_amd64_RSI(%rax), %rsi 126 movq OFFSET_amd64_RDX(%rax), %rdx 127 movq OFFSET_amd64_R10(%rax), %r10 128 movq OFFSET_amd64_R8(%rax), %r8 129 movq OFFSET_amd64_R9(%rax), %r9 130 /* 2 stack parameters. */ 131 movq OFFSET_amd64_RSP(%rax), %rax 132 movq 16(%rax), %r11 133 pushq %r11 134 movq 8(%rax), %r11 135 pushq %r11 136 /* Return address. */ 137 movq 0(%rax), %r11 138 pushq %r11 139 140 /* Put syscall number in %rax. */ 141 movq -48(%rbp), %rax 142 143 /* Do the syscall. Note that the Solaris kernel doesn't directly 144 restart syscalls! */ 145 syscall 146 1472: /* In the range [2, 3), the syscall result is in %rax and %rdx and C, 148 but hasn't been committed to the thread state. If we get 149 interrupted in this section then we'll just use values saved in the 150 ucontext structure. 151 152 Important note for this and the following section: Don't add here 153 any code that alters the carry flag or worse, call any function. 154 That would completely break the fixup after an interrupt. */ 155 movq -40(%rbp), %rcx 156 movq %rax, OFFSET_amd64_RAX(%rcx) /* save %rax to VEX */ 157 movq %rdx, OFFSET_amd64_RDX(%rcx) /* save %rdx to VEX */ 158 movq -16(%rbp), %rcx 159 setc 0(%rcx) /* save returned carry flag */ 160 1613: /* Re-block signals. If %rip is in [3, 4), then the syscall is 162 complete and we do not need to worry about it. We have to only 163 correctly save the carry flag. If we get interrupted in this 164 section then we just have to propagate the carry flag from the 165 ucontext structure to the thread state, %rax and %rdx values are 166 already saved. */ 167 REBLOCK_SIGNALS 168 1694: /* Now safe from signals. */ 170 SIMPLE_RETURN 171 172.section .rodata 173/* Export the ranges so that 174 VG_(fixup_guest_state_after_syscall_interrupted) can do the right thing. */ 175 176.globl ML_(blksys_setup) 177.globl ML_(blksys_complete) 178.globl ML_(blksys_committed) 179.globl ML_(blksys_finished) 180ML_(blksys_setup): .quad 1b 181ML_(blksys_complete): .quad 2b 182ML_(blksys_committed): .quad 3b 183ML_(blksys_finished): .quad 4b 184.previous 185 186/* Prototype: 187 Int ML_(do_syscall_for_client_dret_WRK)( 188 Int syscallno, // %rdi = %rbp-48 = %rbx-48+8 189 void *guest_state, // %rsi = %rbp-40 = %rbx-40+8 190 const vki_sigset_t *sysmask, // %rdx = %rbp-32 = %rbx-32+8 191 const vki_sigset_t *postmask, // %rcx = %rbp-24 = %rbx-24+8 192 UChar *cflag) // %r8 = %rbp-16 = %rbx-16+8 193*/ 194 195/* Door_return is a very special call because the data are stored by the 196 kernel directly on the stack and the stack pointer is appropriately 197 modified by the kernel. Therefore we switch to the client stack before 198 doing the syscall, this is relatively trivial but an extra care has to be 199 taken when we get interrupted at some point. */ 200 201.globl ML_(do_syscall_for_client_dret_WRK) 202ML_(do_syscall_for_client_dret_WRK): 203 ESTABLISH_STACKFRAME 204 2051: /* Even though we can't take a signal until the sigprocmask completes, 206 start the range early. If %rip is in the range [1, 2), the syscall 207 hasn't been started yet. */ 208 UNBLOCK_SIGNALS 209 210 /* Prepare 6 register parameters. */ 211 movq -40(%rbp), %rax 212 movq OFFSET_amd64_RDI(%rax), %rdi 213 movq OFFSET_amd64_RSI(%rax), %rsi 214 movq OFFSET_amd64_RDX(%rax), %rdx 215 movq OFFSET_amd64_R10(%rax), %r10 216 movq OFFSET_amd64_R8(%rax), %r8 217 movq OFFSET_amd64_R9(%rax), %r9 218 219 /* Switch to the client stack. */ 220 movq OFFSET_amd64_RSP(%rax), %rsp /* %rsp = simulated RSP */ 221 /* Change %rbp to a client value. It will always get committed by 222 the fixup code for range [2, 3) so it needs to be set to what the 223 client expects. */ 224 movq OFFSET_amd64_RBP(%rax), %rbp /* %rbp = simulated RBP */ 225 226 /* Put syscall number in %rax. */ 227 movq -48+8(%rbx), %rax 228 229 /* Do the syscall. Note that the Solaris kernel doesn't directly 230 restart syscalls! */ 231 syscall 232 2332: /* In the range [2, 3), the syscall result is in %rax, %rdx, %rsp and 234 %rbp and C, but hasn't been committed to the thread state. If we 235 get interrupted in this section then we'll just use values saved in 236 the ucontext structure. 237 238 Important note for this and the following section: Don't add here 239 any code that alters the carry flag or worse, call any function. 240 That would completely break the fixup after an interrupt. */ 241 movq -40+8(%rbx), %rcx 242 movq %rax, OFFSET_amd64_RAX(%rcx) /* save %rax to VEX */ 243 movq %rdx, OFFSET_amd64_RDX(%rcx) /* save %rdx to VEX */ 244 movq %rsp, OFFSET_amd64_RSP(%rcx) /* save %rsp to VEX */ 245 movq %rbp, OFFSET_amd64_RBP(%rcx) /* save %rbp to VEX */ 246 movq -16+8(%rbx), %rcx 247 setc 0(%rcx) /* save returned carry flag */ 248 249 movq %rbx, %rsp /* switch to V stack */ 250 2513: /* Re-block signals. If %rip is in [3, 4), then the syscall is 252 complete and we do not need worry about it. We have to only 253 correctly save the carry flag. If we get interrupted in this 254 section then we just have to propagate the carry flag from the 255 ucontext structure to the thread state, %rax, %rdx, %rsp and %rbp 256 values are already saved. */ 257 movq %rbx, %rbp 258 addq $8, %rbp 259 REBLOCK_SIGNALS 260 2614: /* Now safe from signals. */ 262 SIMPLE_RETURN 263 264.section .rodata 265.globl ML_(blksys_setup_DRET) 266.globl ML_(blksys_complete_DRET) 267.globl ML_(blksys_committed_DRET) 268.globl ML_(blksys_finished_DRET) 269ML_(blksys_setup_DRET): .quad 1b 270ML_(blksys_complete_DRET): .quad 2b 271ML_(blksys_committed_DRET): .quad 3b 272ML_(blksys_finished_DRET): .quad 4b 273.previous 274 275#endif // defined(VGP_amd64_solaris) 276 277/* Let the linker know we don't need an executable stack */ 278MARK_STACK_NO_EXEC 279 280/*--------------------------------------------------------------------*/ 281/*--- end ---*/ 282/*--------------------------------------------------------------------*/ 283