You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

194 lines
4.8KB

  1. ;*****************************************************************************
  2. ;* Assembly testing and benchmarking tool
  3. ;* Copyright (c) 2008 Loren Merritt
  4. ;* Copyright (c) 2012 Henrik Gramner
  5. ;*
  6. ;* This file is part of Libav.
  7. ;*
  8. ;* Libav is free software; you can redistribute it and/or modify
  9. ;* it under the terms of the GNU General Public License as published by
  10. ;* the Free Software Foundation; either version 2 of the License, or
  11. ;* (at your option) any later version.
  12. ;*
  13. ;* Libav is distributed in the hope that it will be useful,
  14. ;* but WITHOUT ANY WARRANTY; without even the implied warranty of
  15. ;* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  16. ;* GNU General Public License for more details.
  17. ;*
  18. ;* You should have received a copy of the GNU General Public License
  19. ;* along with this program; if not, write to the Free Software
  20. ;* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02111, USA.
  21. ;*****************************************************************************
  22. %define private_prefix checkasm
  23. %include "libavutil/x86/x86inc.asm"
  24. SECTION_RODATA
  25. error_message: db "failed to preserve register", 0
  26. %if ARCH_X86_64
  27. ; just random numbers to reduce the chance of incidental match
  28. ALIGN 16
  29. x6: dq 0x1a1b2550a612b48c,0x79445c159ce79064
  30. x7: dq 0x2eed899d5a28ddcd,0x86b2536fcd8cf636
  31. x8: dq 0xb0856806085e7943,0x3f2bf84fc0fcca4e
  32. x9: dq 0xacbd382dcf5b8de2,0xd229e1f5b281303f
  33. x10: dq 0x71aeaff20b095fd9,0xab63e2e11fa38ed9
  34. x11: dq 0x89b0c0765892729a,0x77d410d5c42c882d
  35. x12: dq 0xc45ea11a955d8dd5,0x24b3c1d2a024048b
  36. x13: dq 0x2e8ec680de14b47c,0xdd7b8919edd42786
  37. x14: dq 0x135ce6888fa02cbf,0x11e53e2b2ac655ef
  38. x15: dq 0x011ff554472a7a10,0x6de8f4c914c334d5
  39. n7: dq 0x21f86d66c8ca00ce
  40. n8: dq 0x75b6ba21077c48ad
  41. n9: dq 0xed56bb2dcb3c7736
  42. n10: dq 0x8bda43d3fd1a7e06
  43. n11: dq 0xb64a9c9e5d318408
  44. n12: dq 0xdf9a54b303f1d3a3
  45. n13: dq 0x4a75479abd64e097
  46. n14: dq 0x249214109d5d1c88
  47. %endif
  48. SECTION .text
  49. cextern fail_func
  50. ; max number of args used by any asm function.
  51. ; (max_args % 4) must equal 3 for stack alignment
  52. %define max_args 15
  53. %if ARCH_X86_64
  54. ;-----------------------------------------------------------------------------
  55. ; int checkasm_stack_clobber(uint64_t clobber, ...)
  56. ;-----------------------------------------------------------------------------
  57. cglobal stack_clobber, 1,2
  58. ; Clobber the stack with junk below the stack pointer
  59. %define size (max_args+6)*8
  60. SUB rsp, size
  61. mov r1, size-8
  62. .loop:
  63. mov [rsp+r1], r0
  64. sub r1, 8
  65. jge .loop
  66. ADD rsp, size
  67. RET
  68. %if WIN64
  69. %assign free_regs 7
  70. %else
  71. %assign free_regs 9
  72. %endif
  73. ;-----------------------------------------------------------------------------
  74. ; intptr_t checkasm_checked_call(intptr_t (*func)(), ...)
  75. ;-----------------------------------------------------------------------------
  76. INIT_XMM
  77. cglobal checked_call, 2,15,16,max_args*8+8
  78. mov r6, r0
  79. ; All arguments have been pushed on the stack instead of registers in order to
  80. ; test for incorrect assumptions that 32-bit ints are zero-extended to 64-bit.
  81. mov r0, r6mp
  82. mov r1, r7mp
  83. mov r2, r8mp
  84. mov r3, r9mp
  85. %if UNIX64
  86. mov r4, r10mp
  87. mov r5, r11mp
  88. %assign i 6
  89. %rep max_args-6
  90. mov r9, [rsp+stack_offset+(i+1)*8]
  91. mov [rsp+(i-6)*8], r9
  92. %assign i i+1
  93. %endrep
  94. %else
  95. %assign i 4
  96. %rep max_args-4
  97. mov r9, [rsp+stack_offset+(i+7)*8]
  98. mov [rsp+i*8], r9
  99. %assign i i+1
  100. %endrep
  101. %endif
  102. %if WIN64
  103. %assign i 6
  104. %rep 16-6
  105. mova m %+ i, [x %+ i]
  106. %assign i i+1
  107. %endrep
  108. %endif
  109. %assign i 14
  110. %rep 15-free_regs
  111. mov r %+ i, [n %+ i]
  112. %assign i i-1
  113. %endrep
  114. call r6
  115. %assign i 14
  116. %rep 15-free_regs
  117. xor r %+ i, [n %+ i]
  118. or r14, r %+ i
  119. %assign i i-1
  120. %endrep
  121. %if WIN64
  122. %assign i 6
  123. %rep 16-6
  124. pxor m %+ i, [x %+ i]
  125. por m6, m %+ i
  126. %assign i i+1
  127. %endrep
  128. packsswb m6, m6
  129. movq r5, m6
  130. or r14, r5
  131. %endif
  132. jz .ok
  133. mov r9, rax
  134. lea r0, [error_message]
  135. call fail_func
  136. mov rax, r9
  137. .ok:
  138. RET
  139. %else
  140. ; just random numbers to reduce the chance of incidental match
  141. %define n3 dword 0x6549315c
  142. %define n4 dword 0xe02f3e23
  143. %define n5 dword 0xb78d0d1d
  144. %define n6 dword 0x33627ba7
  145. ;-----------------------------------------------------------------------------
  146. ; intptr_t checkasm_checked_call(intptr_t (*func)(), ...)
  147. ;-----------------------------------------------------------------------------
  148. cglobal checked_call, 1,7
  149. mov r3, n3
  150. mov r4, n4
  151. mov r5, n5
  152. mov r6, n6
  153. %rep max_args
  154. PUSH dword [esp+20+max_args*4]
  155. %endrep
  156. call r0
  157. xor r3, n3
  158. xor r4, n4
  159. xor r5, n5
  160. xor r6, n6
  161. or r3, r4
  162. or r5, r6
  163. or r3, r5
  164. jz .ok
  165. mov r3, eax
  166. lea r0, [error_message]
  167. mov [esp], r0
  168. call fail_func
  169. mov eax, r3
  170. .ok:
  171. add esp, max_args*4
  172. REP_RET
  173. %endif ; ARCH_X86_64