mc-a2.asm
上传用户:lctgjx
上传日期:2022-06-04
资源大小:8887k
文件大小:26k
源码类别:

流媒体/Mpeg4/MP4

开发平台:

Visual C++

  1. ;*****************************************************************************
  2. ;* mc-a2.asm: h264 encoder library
  3. ;*****************************************************************************
  4. ;* Copyright (C) 2005-2008 x264 project
  5. ;*
  6. ;* Authors: Loren Merritt <lorenm@u.washington.edu>
  7. ;*          Jason Garrett-Glaser <darkshikari@gmail.com>
  8. ;*          Holger Lubitz <hal@duncan.ol.sub.de>
  9. ;*          Mathieu Monnier <manao@melix.net>
  10. ;*
  11. ;* This program is free software; you can redistribute it and/or modify
  12. ;* it under the terms of the GNU General Public License as published by
  13. ;* the Free Software Foundation; either version 2 of the License, or
  14. ;* (at your option) any later version.
  15. ;*
  16. ;* This program is distributed in the hope that it will be useful,
  17. ;* but WITHOUT ANY WARRANTY; without even the implied warranty of
  18. ;* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  19. ;* GNU General Public License for more details.
  20. ;*
  21. ;* You should have received a copy of the GNU General Public License
  22. ;* along with this program; if not, write to the Free Software
  23. ;* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02111, USA.
  24. ;*****************************************************************************
  25. %include "x86inc.asm"
  26. %include "x86util.asm"
  27. SECTION_RODATA
  28. filt_mul20: times 16 db 20
  29. filt_mul51: times 8 db 1, -5
  30. pw_1:  times 8 dw 1
  31. pw_16: times 8 dw 16
  32. pw_32: times 8 dw 32
  33. pd_128: times 4 dd 128
  34. SECTION .text
  35. %macro LOAD_ADD 4
  36.     movh       %4, %3
  37.     movh       %1, %2
  38.     punpcklbw  %4, m0
  39.     punpcklbw  %1, m0
  40.     paddw      %1, %4
  41. %endmacro
  42. %macro LOAD_ADD_2 6
  43.     mova       %5, %3
  44.     mova       %1, %4
  45.     mova       %6, %5
  46.     mova       %2, %1
  47.     punpcklbw  %5, m0
  48.     punpcklbw  %1, m0
  49.     punpckhbw  %6, m0
  50.     punpckhbw  %2, m0
  51.     paddw      %1, %5
  52.     paddw      %2, %6
  53. %endmacro
  54. %macro FILT_V2 0
  55.     psubw  m1, m2  ; a-b
  56.     psubw  m4, m5
  57.     psubw  m2, m3  ; b-c
  58.     psubw  m5, m6
  59.     psllw  m2, 2
  60.     psllw  m5, 2
  61.     psubw  m1, m2  ; a-5*b+4*c
  62.     psubw  m4, m5
  63.     psllw  m3, 4
  64.     psllw  m6, 4
  65.     paddw  m1, m3  ; a-5*b+20*c
  66.     paddw  m4, m6
  67. %endmacro
  68. %macro FILT_H 3
  69.     psubw  %1, %2  ; a-b
  70.     psraw  %1, 2   ; (a-b)/4
  71.     psubw  %1, %2  ; (a-b)/4-b
  72.     paddw  %1, %3  ; (a-b)/4-b+c
  73.     psraw  %1, 2   ; ((a-b)/4-b+c)/4
  74.     paddw  %1, %3  ; ((a-b)/4-b+c)/4+c = (a-5*b+20*c)/16
  75. %endmacro
  76. %macro FILT_H2 6
  77.     psubw  %1, %2
  78.     psubw  %4, %5
  79.     psraw  %1, 2
  80.     psraw  %4, 2
  81.     psubw  %1, %2
  82.     psubw  %4, %5
  83.     paddw  %1, %3
  84.     paddw  %4, %6
  85.     psraw  %1, 2
  86.     psraw  %4, 2
  87.     paddw  %1, %3
  88.     paddw  %4, %6
  89. %endmacro
  90. %macro FILT_PACK 3
  91.     paddw     %1, m7
  92.     paddw     %2, m7
  93.     psraw     %1, %3
  94.     psraw     %2, %3
  95.     packuswb  %1, %2
  96. %endmacro
  97. INIT_MMX
  98. %macro HPEL_V 1-2 0
  99. ;-----------------------------------------------------------------------------
  100. ; void x264_hpel_filter_v_mmxext( uint8_t *dst, uint8_t *src, int16_t *buf, int stride, int width );
  101. ;-----------------------------------------------------------------------------
  102. cglobal x264_hpel_filter_v_%1, 5,6,%2
  103. %ifdef WIN64
  104.     movsxd   r4, r4d
  105. %endif
  106.     lea r5, [r1+r3]
  107.     sub r1, r3
  108.     sub r1, r3
  109.     add r0, r4
  110.     lea r2, [r2+r4*2]
  111.     neg r4
  112. %ifnidn %1, ssse3
  113.     pxor m0, m0
  114. %else
  115.     mova m0, [filt_mul51 GLOBAL]
  116. %endif
  117. .loop:
  118. %ifidn %1, ssse3
  119.     mova m1, [r1]
  120.     mova m4, [r1+r3]
  121.     mova m2, [r5+r3*2]
  122.     mova m5, [r5+r3]
  123.     mova m3, [r1+r3*2]
  124.     mova m6, [r5]
  125.     SBUTTERFLY bw, 1, 4, 7
  126.     SBUTTERFLY bw, 2, 5, 7
  127.     SBUTTERFLY bw, 3, 6, 7
  128.     pmaddubsw m1, m0
  129.     pmaddubsw m4, m0
  130.     pmaddubsw m2, m0
  131.     pmaddubsw m5, m0
  132.     pmaddubsw m3, [filt_mul20 GLOBAL]
  133.     pmaddubsw m6, [filt_mul20 GLOBAL]
  134.     paddw  m1, m2
  135.     paddw  m4, m5
  136.     paddw  m1, m3
  137.     paddw  m4, m6
  138. %else
  139.     LOAD_ADD_2 m1, m4, [r1     ], [r5+r3*2], m6, m7            ; a0 / a1
  140.     LOAD_ADD_2 m2, m5, [r1+r3  ], [r5+r3  ], m6, m7            ; b0 / b1
  141.     LOAD_ADD   m3,     [r1+r3*2], [r5     ], m7                ; c0
  142.     LOAD_ADD   m6,     [r1+r3*2+mmsize/2], [r5+mmsize/2], m7 ; c1
  143.     FILT_V2
  144. %endif
  145.     mova      m7, [pw_16 GLOBAL]
  146.     mova      [r2+r4*2], m1
  147.     mova      [r2+r4*2+mmsize], m4
  148.     paddw     m1, m7
  149.     paddw     m4, m7
  150.     psraw     m1, 5
  151.     psraw     m4, 5
  152.     packuswb  m1, m4
  153.     mova     [r0+r4], m1
  154.     add r1, mmsize
  155.     add r5, mmsize
  156.     add r4, mmsize
  157.     jl .loop
  158.     REP_RET
  159. %endmacro
  160. HPEL_V mmxext
  161. ;-----------------------------------------------------------------------------
  162. ; void x264_hpel_filter_c_mmxext( uint8_t *dst, int16_t *buf, int width );
  163. ;-----------------------------------------------------------------------------
  164. cglobal x264_hpel_filter_c_mmxext, 3,3
  165.     add r0, r2
  166.     lea r1, [r1+r2*2]
  167.     neg r2
  168.     %define src r1+r2*2
  169.     movq m7, [pw_32 GLOBAL]
  170. .loop:
  171.     movq   m1, [src-4]
  172.     movq   m2, [src-2]
  173.     movq   m3, [src  ]
  174.     movq   m4, [src+4]
  175.     movq   m5, [src+6]
  176.     paddw  m3, [src+2]  ; c0
  177.     paddw  m2, m4       ; b0
  178.     paddw  m1, m5       ; a0
  179.     movq   m6, [src+8]
  180.     paddw  m4, [src+14] ; a1
  181.     paddw  m5, [src+12] ; b1
  182.     paddw  m6, [src+10] ; c1
  183.     FILT_H2 m1, m2, m3, m4, m5, m6
  184.     FILT_PACK m1, m4, 6
  185.     movntq [r0+r2], m1
  186.     add r2, 8
  187.     jl .loop
  188.     REP_RET
  189. ;-----------------------------------------------------------------------------
  190. ; void x264_hpel_filter_h_mmxext( uint8_t *dst, uint8_t *src, int width );
  191. ;-----------------------------------------------------------------------------
  192. cglobal x264_hpel_filter_h_mmxext, 3,3
  193.     add r0, r2
  194.     add r1, r2
  195.     neg r2
  196.     %define src r1+r2
  197.     pxor m0, m0
  198. .loop:
  199.     movd       m1, [src-2]
  200.     movd       m2, [src-1]
  201.     movd       m3, [src  ]
  202.     movd       m6, [src+1]
  203.     movd       m4, [src+2]
  204.     movd       m5, [src+3]
  205.     punpcklbw  m1, m0
  206.     punpcklbw  m2, m0
  207.     punpcklbw  m3, m0
  208.     punpcklbw  m6, m0
  209.     punpcklbw  m4, m0
  210.     punpcklbw  m5, m0
  211.     paddw      m3, m6 ; c0
  212.     paddw      m2, m4 ; b0
  213.     paddw      m1, m5 ; a0
  214.     movd       m7, [src+7]
  215.     movd       m6, [src+6]
  216.     punpcklbw  m7, m0
  217.     punpcklbw  m6, m0
  218.     paddw      m4, m7 ; c1
  219.     paddw      m5, m6 ; b1
  220.     movd       m7, [src+5]
  221.     movd       m6, [src+4]
  222.     punpcklbw  m7, m0
  223.     punpcklbw  m6, m0
  224.     paddw      m6, m7 ; a1
  225.     movq       m7, [pw_1 GLOBAL]
  226.     FILT_H2 m1, m2, m3, m4, m5, m6
  227.     FILT_PACK m1, m4, 1
  228.     movntq     [r0+r2], m1
  229.     add r2, 8
  230.     jl .loop
  231.     REP_RET
  232. INIT_XMM
  233. %macro HPEL_C 1
  234. ;-----------------------------------------------------------------------------
  235. ; void x264_hpel_filter_c_sse2( uint8_t *dst, int16_t *buf, int width );
  236. ;-----------------------------------------------------------------------------
  237. cglobal x264_hpel_filter_c_%1, 3,3,9
  238.     add r0, r2
  239.     lea r1, [r1+r2*2]
  240.     neg r2
  241.     %define src r1+r2*2
  242. %ifidn %1, ssse3
  243.     mova    m7, [pw_32 GLOBAL]
  244.     %define tpw_32 m7
  245. %elifdef ARCH_X86_64
  246.     mova    m8, [pw_32 GLOBAL]
  247.     %define tpw_32 m8
  248. %else
  249.     %define tpw_32 [pw_32 GLOBAL]
  250. %endif
  251. .loop:
  252. %ifidn %1,sse2_misalign
  253.     movu    m0, [src-4]
  254.     movu    m1, [src-2]
  255.     mova    m2, [src]
  256.     paddw   m0, [src+6]
  257.     paddw   m1, [src+4]
  258.     paddw   m2, [src+2]
  259. %else
  260.     mova    m6, [src-16]
  261.     mova    m2, [src]
  262.     mova    m3, [src+16]
  263.     mova    m0, m2
  264.     mova    m1, m2
  265.     mova    m4, m3
  266.     mova    m5, m3
  267.     PALIGNR m3, m2, 2, m7
  268.     PALIGNR m4, m2, 4, m7
  269.     PALIGNR m5, m2, 6, m7
  270.     PALIGNR m0, m6, 12, m7
  271.     PALIGNR m1, m6, 14, m7
  272.     paddw   m2, m3
  273.     paddw   m1, m4
  274.     paddw   m0, m5
  275. %endif
  276.     FILT_H  m0, m1, m2
  277.     paddw   m0, tpw_32
  278.     psraw   m0, 6
  279.     packuswb m0, m0
  280.     movq [r0+r2], m0
  281.     add r2, 8
  282.     jl .loop
  283.     REP_RET
  284. %endmacro
  285. ;-----------------------------------------------------------------------------
  286. ; void x264_hpel_filter_h_sse2( uint8_t *dst, uint8_t *src, int width );
  287. ;-----------------------------------------------------------------------------
  288. cglobal x264_hpel_filter_h_sse2, 3,3,8
  289.     add r0, r2
  290.     add r1, r2
  291.     neg r2
  292.     %define src r1+r2
  293.     pxor m0, m0
  294. .loop:
  295.     movh       m1, [src-2]
  296.     movh       m2, [src-1]
  297.     movh       m3, [src  ]
  298.     movh       m4, [src+1]
  299.     movh       m5, [src+2]
  300.     movh       m6, [src+3]
  301.     punpcklbw  m1, m0
  302.     punpcklbw  m2, m0
  303.     punpcklbw  m3, m0
  304.     punpcklbw  m4, m0
  305.     punpcklbw  m5, m0
  306.     punpcklbw  m6, m0
  307.     paddw      m3, m4 ; c0
  308.     paddw      m2, m5 ; b0
  309.     paddw      m1, m6 ; a0
  310.     movh       m4, [src+6]
  311.     movh       m5, [src+7]
  312.     movh       m6, [src+10]
  313.     movh       m7, [src+11]
  314.     punpcklbw  m4, m0
  315.     punpcklbw  m5, m0
  316.     punpcklbw  m6, m0
  317.     punpcklbw  m7, m0
  318.     paddw      m5, m6 ; b1
  319.     paddw      m4, m7 ; a1
  320.     movh       m6, [src+8]
  321.     movh       m7, [src+9]
  322.     punpcklbw  m6, m0
  323.     punpcklbw  m7, m0
  324.     paddw      m6, m7 ; c1
  325.     mova       m7, [pw_1 GLOBAL] ; FIXME xmm8
  326.     FILT_H2 m1, m2, m3, m4, m5, m6
  327.     FILT_PACK m1, m4, 1
  328.     movntdq    [r0+r2], m1
  329.     add r2, 16
  330.     jl .loop
  331.     REP_RET
  332. %ifndef ARCH_X86_64
  333. ;-----------------------------------------------------------------------------
  334. ; void x264_hpel_filter_h_ssse3( uint8_t *dst, uint8_t *src, int width );
  335. ;-----------------------------------------------------------------------------
  336. cglobal x264_hpel_filter_h_ssse3, 3,3
  337.     add r0, r2
  338.     add r1, r2
  339.     neg r2
  340.     %define src r1+r2
  341.     pxor m0, m0
  342.     movh m1, [src-8]
  343.     punpcklbw m1, m0         ; 00 -1 00 -2 00 -3 00 -4 00 -5 00 -6 00 -7 00 -8
  344.     movh m2, [src]
  345.     punpcklbw m2, m0
  346.     mova       m7, [pw_1 GLOBAL]
  347. .loop:
  348.     movh       m3, [src+8]
  349.     punpcklbw  m3, m0
  350.     mova       m4, m2
  351.     palignr    m2, m1, 14
  352.     mova       m5, m3
  353.     palignr    m3, m4, 4
  354.     paddw      m3, m2
  355.     mova       m2, m4
  356.     palignr    m4, m1, 12
  357.     mova       m1, m5
  358.     palignr    m5, m2, 6
  359.     paddw      m5, m4
  360.     mova       m4, m1
  361.     palignr    m1, m2, 2
  362.     paddw      m1, m2
  363.     FILT_H     m5, m3, m1
  364.     movh       m1, [src+16]
  365.     punpcklbw  m1, m0
  366.     mova       m3, m4
  367.     palignr    m4, m2, 14
  368.     mova       m6, m1
  369.     palignr    m1, m3, 4
  370.     paddw      m1, m4
  371.     mova       m4, m3
  372.     palignr    m3, m2, 12
  373.     mova       m2, m6
  374.     palignr    m6, m4, 6
  375.     paddw      m6, m3
  376.     mova       m3, m2
  377.     palignr    m2, m4, 2
  378.     paddw      m2, m4
  379.     FILT_H m6, m1, m2
  380.     FILT_PACK m5, m6, 1
  381.     movdqa    [r0+r2], m5
  382.     add r2, 16
  383.     mova      m2, m3
  384.     mova      m1, m4
  385.     jl .loop
  386.     REP_RET
  387. %endif
  388. %define PALIGNR PALIGNR_MMX
  389. %ifndef ARCH_X86_64
  390. HPEL_C sse2
  391. %endif
  392. HPEL_V sse2, 8
  393. HPEL_C sse2_misalign
  394. %define PALIGNR PALIGNR_SSSE3
  395. HPEL_C ssse3
  396. HPEL_V ssse3
  397. %ifdef ARCH_X86_64
  398. %macro DO_FILT_V 6
  399. %ifidn %6, ssse3
  400.     mova m1, [r3]
  401.     mova m2, [r3+r2]
  402.     mova %3, [r3+r2*2]
  403.     mova m3, [r1]
  404.     mova %4, [r1+r2]
  405.     mova m0, [r1+r2*2]
  406.     mova %2, [filt_mul51 GLOBAL]
  407.     mova m4, m1
  408.     punpcklbw m1, m2
  409.     punpckhbw m4, m2
  410.     mova m2, m0
  411.     punpcklbw m0, %4
  412.     punpckhbw m2, %4
  413.     mova %1, m3
  414.     punpcklbw m3, %3
  415.     punpckhbw %1, %3
  416.     mova %3, m3
  417.     mova %4, %1
  418.     pmaddubsw m1, %2
  419.     pmaddubsw m4, %2
  420.     pmaddubsw m0, %2
  421.     pmaddubsw m2, %2
  422.     pmaddubsw m3, [filt_mul20 GLOBAL]
  423.     pmaddubsw %1, [filt_mul20 GLOBAL]
  424.     psrlw     %3, 8
  425.     psrlw     %4, 8
  426.     paddw m1, m0
  427.     paddw m4, m2
  428.     paddw m1, m3
  429.     paddw m4, %1
  430. %else
  431.     LOAD_ADD_2 m1, m4, [r3     ], [r1+r2*2], m2, m5            ; a0 / a1
  432.     LOAD_ADD_2 m2, m5, [r3+r2  ], [r1+r2  ], m3, m6            ; b0 / b1
  433.     LOAD_ADD_2 m3, m6, [r3+r2*2], [r1     ], %3, %4            ; c0 / c1
  434.     FILT_V2
  435. %endif
  436.     mova      %1, m1
  437.     mova      %2, m4
  438.     paddw     m1, m15
  439.     paddw     m4, m15
  440.     add       r3, 16
  441.     add       r1, 16
  442.     psraw     m1, 5
  443.     psraw     m4, 5
  444.     packuswb  m1, m4
  445.     movntps  [r11+r4+%5], m1
  446. %endmacro
  447. %macro DO_FILT_H 4
  448.     mova      m1, %2
  449.     PALIGNR   m1, %1, 12, m4
  450.     mova      m2, %2
  451.     PALIGNR   m2, %1, 14, m4
  452.     mova      %1, %3
  453.     PALIGNR   %3, %2, 6, m4
  454.     mova      m3, %1
  455.     PALIGNR   m3, %2, 4, m4
  456.     mova      m4, %1
  457.     paddw     %3, m1
  458.     PALIGNR   m4, %2, 2, m1
  459.     paddw     m3, m2
  460.     paddw     m4, %2
  461.     FILT_H    %3, m3, m4
  462.     paddw     %3, m15
  463.     psraw     %3, %4
  464. %endmacro
  465. %macro DO_FILT_CC 4
  466.     DO_FILT_H %1, %2, %3, 6
  467.     DO_FILT_H %2, %1, %4, 6
  468.     packuswb  %3, %4
  469.     movntps   [r5+r4], %3
  470. %endmacro
  471. %macro DO_FILT_HH 4
  472.     DO_FILT_H %1, %2, %3, 1
  473.     DO_FILT_H %2, %1, %4, 1
  474.     packuswb  %3, %4
  475.     movntps   [r0+r4], %3
  476. %endmacro
  477. %macro DO_FILT_H2 6
  478.     DO_FILT_H %1, %2, %3, 6
  479.     psrlw    m15, 5
  480.     DO_FILT_H %4, %5, %6, 1
  481.     packuswb  %6, %3
  482. %endmacro
  483. %macro HPEL 1
  484. ;-----------------------------------------------------------------------------
  485. ; void x264_hpel_filter_sse2( uint8_t *dsth, uint8_t *dstv, uint8_t *dstc,
  486. ;                             uint8_t *src, int stride, int width, int height)
  487. ;-----------------------------------------------------------------------------
  488. cglobal x264_hpel_filter_%1, 7,7,16
  489. %ifdef WIN64
  490.     movsxd   r4, r4d
  491.     movsxd   r5, r5d
  492. %endif
  493.     mov      r10, r3
  494.     sub       r5, 16
  495.     mov      r11, r1
  496.     and      r10, 15
  497.     sub       r3, r10
  498.     add       r0, r5
  499.     add      r11, r5
  500.     add      r10, r5
  501.     add       r5, r2
  502.     mov       r2, r4
  503.     neg      r10
  504.     lea       r1, [r3+r2]
  505.     sub       r3, r2
  506.     sub       r3, r2
  507.     mov       r4, r10
  508. %ifidn %1, sse2
  509.     pxor      m0, m0
  510. %endif
  511.     pcmpeqw  m15, m15
  512.     psrlw    m15, 15 ; pw_1
  513.     psllw    m15, 4
  514. ;ALIGN 16
  515. .loopy:
  516. ; first filter_v
  517. ; prefetching does not help here! lots of variants tested, all slower
  518.     DO_FILT_V m8, m7, m13, m12, 0, %1
  519. ;ALIGN 16
  520. .loopx:
  521.     DO_FILT_V m6, m5, m11, m10, 16, %1
  522. .lastx:
  523.     paddw    m15, m15
  524.     DO_FILT_CC m9, m8, m7, m6
  525.     movdqa   m7, m12        ; not really necessary, but seems free and
  526.     movdqa   m6, m11      ; gives far shorter code
  527.     psrlw    m15, 5
  528.     DO_FILT_HH m14, m13, m7, m6
  529.     psllw    m15, 4 ; pw_16
  530.     movdqa   m7, m5
  531.     movdqa  m12, m10
  532.     add      r4, 16
  533.     jl .loopx
  534.     cmp      r4, 16
  535.     jl .lastx
  536. ; setup regs for next y
  537.     sub      r4, r10
  538.     sub      r4, r2
  539.     sub      r1, r4
  540.     sub      r3, r4
  541.     add      r0, r2
  542.     add     r11, r2
  543.     add      r5, r2
  544.     mov      r4, r10
  545.     sub     r6d, 1
  546.     jg .loopy
  547.     sfence
  548.     RET
  549. %endmacro
  550. %define PALIGNR PALIGNR_MMX
  551. HPEL sse2
  552. %define PALIGNR PALIGNR_SSSE3
  553. HPEL ssse3
  554. %endif
  555. cglobal x264_sfence
  556.     sfence
  557.     ret
  558. ;-----------------------------------------------------------------------------
  559. ; void x264_plane_copy_mmxext( uint8_t *dst, int i_dst,
  560. ;                              uint8_t *src, int i_src, int w, int h)
  561. ;-----------------------------------------------------------------------------
  562. cglobal x264_plane_copy_mmxext, 6,7
  563.     movsxdifnidn r1, r1d
  564.     movsxdifnidn r3, r3d
  565.     add    r4d, 3
  566.     and    r4d, ~3
  567.     mov    r6d, r4d
  568.     and    r6d, ~15
  569.     sub    r1,  r6
  570.     sub    r3,  r6
  571. .loopy:
  572.     mov    r6d, r4d
  573.     sub    r6d, 64
  574.     jl     .endx
  575. .loopx:
  576.     prefetchnta [r2+256]
  577.     movq   mm0, [r2   ]
  578.     movq   mm1, [r2+ 8]
  579.     movq   mm2, [r2+16]
  580.     movq   mm3, [r2+24]
  581.     movq   mm4, [r2+32]
  582.     movq   mm5, [r2+40]
  583.     movq   mm6, [r2+48]
  584.     movq   mm7, [r2+56]
  585.     movntq [r0   ], mm0
  586.     movntq [r0+ 8], mm1
  587.     movntq [r0+16], mm2
  588.     movntq [r0+24], mm3
  589.     movntq [r0+32], mm4
  590.     movntq [r0+40], mm5
  591.     movntq [r0+48], mm6
  592.     movntq [r0+56], mm7
  593.     add    r2,  64
  594.     add    r0,  64
  595.     sub    r6d, 64
  596.     jge    .loopx
  597. .endx:
  598.     prefetchnta [r2+256]
  599.     add    r6d, 48
  600.     jl .end16
  601. .loop16:
  602.     movq   mm0, [r2  ]
  603.     movq   mm1, [r2+8]
  604.     movntq [r0  ], mm0
  605.     movntq [r0+8], mm1
  606.     add    r2,  16
  607.     add    r0,  16
  608.     sub    r6d, 16
  609.     jge    .loop16
  610. .end16:
  611.     add    r6d, 12
  612.     jl .end4
  613. .loop4:
  614.     movd   mm2, [r2+r6]
  615.     movd   [r0+r6], mm2
  616.     sub    r6d, 4
  617.     jge .loop4
  618. .end4:
  619.     add    r2, r3
  620.     add    r0, r1
  621.     dec    r5d
  622.     jg     .loopy
  623.     sfence
  624.     emms
  625.     RET
  626. ; These functions are not general-use; not only do the SSE ones require aligned input,
  627. ; but they also will fail if given a non-mod16 size or a size less than 64.
  628. ; memzero SSE will fail for non-mod128.
  629. ;-----------------------------------------------------------------------------
  630. ; void *x264_memcpy_aligned_mmx( void *dst, const void *src, size_t n );
  631. ;-----------------------------------------------------------------------------
  632. cglobal x264_memcpy_aligned_mmx, 3,3
  633.     test r2d, 16
  634.     jz .copy32
  635.     sub r2d, 16
  636.     movq mm0, [r1 + r2 + 0]
  637.     movq mm1, [r1 + r2 + 8]
  638.     movq [r0 + r2 + 0], mm0
  639.     movq [r0 + r2 + 8], mm1
  640. .copy32:
  641.     sub r2d, 32
  642.     movq mm0, [r1 + r2 +  0]
  643.     movq mm1, [r1 + r2 +  8]
  644.     movq mm2, [r1 + r2 + 16]
  645.     movq mm3, [r1 + r2 + 24]
  646.     movq [r0 + r2 +  0], mm0
  647.     movq [r0 + r2 +  8], mm1
  648.     movq [r0 + r2 + 16], mm2
  649.     movq [r0 + r2 + 24], mm3
  650.     jg .copy32
  651.     REP_RET
  652. ;-----------------------------------------------------------------------------
  653. ; void *x264_memcpy_aligned_sse2( void *dst, const void *src, size_t n );
  654. ;-----------------------------------------------------------------------------
  655. cglobal x264_memcpy_aligned_sse2, 3,3
  656.     test r2d, 16
  657.     jz .copy32
  658.     sub r2d, 16
  659.     movdqa xmm0, [r1 + r2]
  660.     movdqa [r0 + r2], xmm0
  661. .copy32:
  662.     test r2d, 32
  663.     jz .copy64
  664.     sub r2d, 32
  665.     movdqa xmm0, [r1 + r2 +  0]
  666.     movdqa [r0 + r2 +  0], xmm0
  667.     movdqa xmm1, [r1 + r2 + 16]
  668.     movdqa [r0 + r2 + 16], xmm1
  669. .copy64:
  670.     sub r2d, 64
  671.     movdqa xmm0, [r1 + r2 +  0]
  672.     movdqa [r0 + r2 +  0], xmm0
  673.     movdqa xmm1, [r1 + r2 + 16]
  674.     movdqa [r0 + r2 + 16], xmm1
  675.     movdqa xmm2, [r1 + r2 + 32]
  676.     movdqa [r0 + r2 + 32], xmm2
  677.     movdqa xmm3, [r1 + r2 + 48]
  678.     movdqa [r0 + r2 + 48], xmm3
  679.     jg .copy64
  680.     REP_RET
  681. ;-----------------------------------------------------------------------------
  682. ; void *x264_memzero_aligned( void *dst, size_t n );
  683. ;-----------------------------------------------------------------------------
  684. %macro MEMZERO 1
  685. cglobal x264_memzero_aligned_%1, 2,2
  686.     pxor m0, m0
  687. .loop:
  688.     sub r1d, mmsize*8
  689. %assign i 0
  690. %rep 8
  691.     mova [r0 + r1 + i], m0
  692. %assign i i+mmsize
  693. %endrep
  694.     jg .loop
  695.     REP_RET
  696. %endmacro
  697. INIT_MMX
  698. MEMZERO mmx
  699. INIT_XMM
  700. MEMZERO sse2
  701. ;-----------------------------------------------------------------------------
  702. ; void x264_integral_init4h_sse4( uint16_t *sum, uint8_t *pix, int stride )
  703. ;-----------------------------------------------------------------------------
  704. cglobal x264_integral_init4h_sse4, 3,4
  705.     lea     r3, [r0+r2*2]
  706.     add     r1, r2
  707.     neg     r2
  708.     pxor    m4, m4
  709. .loop:
  710.     movdqa  m0, [r1+r2]
  711.     movdqa  m1, [r1+r2+16]
  712.     palignr m1, m0, 8
  713.     mpsadbw m0, m4, 0
  714.     mpsadbw m1, m4, 0
  715.     paddw   m0, [r0+r2*2]
  716.     paddw   m1, [r0+r2*2+16]
  717.     movdqa  [r3+r2*2   ], m0
  718.     movdqa  [r3+r2*2+16], m1
  719.     add     r2, 16
  720.     jl .loop
  721.     REP_RET
  722. cglobal x264_integral_init8h_sse4, 3,4
  723.     lea     r3, [r0+r2*2]
  724.     add     r1, r2
  725.     neg     r2
  726.     pxor    m4, m4
  727. .loop:
  728.     movdqa  m0, [r1+r2]
  729.     movdqa  m1, [r1+r2+16]
  730.     palignr m1, m0, 8
  731.     movdqa  m2, m0
  732.     movdqa  m3, m1
  733.     mpsadbw m0, m4, 0
  734.     mpsadbw m1, m4, 0
  735.     mpsadbw m2, m4, 4
  736.     mpsadbw m3, m4, 4
  737.     paddw   m0, [r0+r2*2]
  738.     paddw   m1, [r0+r2*2+16]
  739.     paddw   m0, m2
  740.     paddw   m1, m3
  741.     movdqa  [r3+r2*2   ], m0
  742.     movdqa  [r3+r2*2+16], m1
  743.     add     r2, 16
  744.     jl .loop
  745.     REP_RET
  746. %macro INTEGRAL_INIT_8V 1
  747. ;-----------------------------------------------------------------------------
  748. ; void x264_integral_init8v_mmx( uint16_t *sum8, int stride )
  749. ;-----------------------------------------------------------------------------
  750. cglobal x264_integral_init8v_%1, 3,3
  751.     shl   r1, 1
  752.     add   r0, r1
  753.     lea   r2, [r0+r1*8]
  754.     neg   r1
  755. .loop:
  756.     mova  m0, [r2+r1]
  757.     mova  m1, [r2+r1+mmsize]
  758.     psubw m0, [r0+r1]
  759.     psubw m1, [r0+r1+mmsize]
  760.     mova  [r0+r1], m0
  761.     mova  [r0+r1+mmsize], m1
  762.     add   r1, 2*mmsize
  763.     jl .loop
  764.     REP_RET
  765. %endmacro
  766. INIT_MMX
  767. INTEGRAL_INIT_8V mmx
  768. INIT_XMM
  769. INTEGRAL_INIT_8V sse2
  770. ;-----------------------------------------------------------------------------
  771. ; void x264_integral_init4v_mmx( uint16_t *sum8, uint16_t *sum4, int stride )
  772. ;-----------------------------------------------------------------------------
  773. INIT_MMX
  774. cglobal x264_integral_init4v_mmx, 3,5
  775.     shl   r2, 1
  776.     lea   r3, [r0+r2*4]
  777.     lea   r4, [r0+r2*8]
  778.     mova  m0, [r0+r2]
  779.     mova  m4, [r4+r2]
  780. .loop:
  781.     sub   r2, 8
  782.     mova  m1, m4
  783.     psubw m1, m0
  784.     mova  m4, [r4+r2]
  785.     mova  m0, [r0+r2]
  786.     paddw m1, m4
  787.     mova  m3, [r3+r2]
  788.     psubw m1, m0
  789.     psubw m3, m0
  790.     mova  [r0+r2], m1
  791.     mova  [r1+r2], m3
  792.     jge .loop
  793.     REP_RET
  794. INIT_XMM
  795. cglobal x264_integral_init4v_sse2, 3,5
  796.     shl     r2, 1
  797.     add     r0, r2
  798.     add     r1, r2
  799.     lea     r3, [r0+r2*4]
  800.     lea     r4, [r0+r2*8]
  801.     neg     r2
  802. .loop:
  803.     mova    m0, [r0+r2]
  804.     mova    m1, [r4+r2]
  805.     mova    m2, m0
  806.     mova    m4, m1
  807.     shufpd  m0, [r0+r2+16], 1
  808.     shufpd  m1, [r4+r2+16], 1
  809.     paddw   m0, m2
  810.     paddw   m1, m4
  811.     mova    m3, [r3+r2]
  812.     psubw   m1, m0
  813.     psubw   m3, m2
  814.     mova  [r0+r2], m1
  815.     mova  [r1+r2], m3
  816.     add     r2, 16
  817.     jl .loop
  818.     REP_RET
  819. cglobal x264_integral_init4v_ssse3, 3,5
  820.     shl     r2, 1
  821.     add     r0, r2
  822.     add     r1, r2
  823.     lea     r3, [r0+r2*4]
  824.     lea     r4, [r0+r2*8]
  825.     neg     r2
  826. .loop:
  827.     mova    m2, [r0+r2]
  828.     mova    m0, [r0+r2+16]
  829.     mova    m4, [r4+r2]
  830.     mova    m1, [r4+r2+16]
  831.     palignr m0, m2, 8
  832.     palignr m1, m4, 8
  833.     paddw   m0, m2
  834.     paddw   m1, m4
  835.     mova    m3, [r3+r2]
  836.     psubw   m1, m0
  837.     psubw   m3, m2
  838.     mova  [r0+r2], m1
  839.     mova  [r1+r2], m3
  840.     add     r2, 16
  841.     jl .loop
  842.     REP_RET
  843. %macro FILT8x4 7
  844.     mova      %3, [r0+%7]
  845.     mova      %4, [r0+r5+%7]
  846.     pavgb     %3, %4
  847.     pavgb     %4, [r0+r5*2+%7]
  848.     PALIGNR   %1, %3, 1, m6
  849.     PALIGNR   %2, %4, 1, m6
  850.     pavgb     %1, %3
  851.     pavgb     %2, %4
  852.     mova      %5, %1
  853.     mova      %6, %2
  854.     pand      %1, m7
  855.     pand      %2, m7
  856.     psrlw     %5, 8
  857.     psrlw     %6, 8
  858. %endmacro
  859. %macro FILT16x2 4
  860.     mova      m3, [r0+%4+mmsize]
  861.     mova      m2, [r0+%4]
  862.     pavgb     m3, [r0+%4+r5+mmsize]
  863.     pavgb     m2, [r0+%4+r5]
  864.     PALIGNR   %1, m3, 1, m6
  865.     pavgb     %1, m3
  866.     PALIGNR   m3, m2, 1, m6
  867.     pavgb     m3, m2
  868.     mova      m5, m3
  869.     mova      m4, %1
  870.     pand      m3, m7
  871.     pand      %1, m7
  872.     psrlw     m5, 8
  873.     psrlw     m4, 8
  874.     packuswb  m3, %1
  875.     packuswb  m5, m4
  876.     mova    [%2], m3
  877.     mova    [%3], m5
  878.     mova      %1, m2
  879. %endmacro
  880. %macro FILT8x2U 3
  881.     mova      m3, [r0+%3+8]
  882.     mova      m2, [r0+%3]
  883.     pavgb     m3, [r0+%3+r5+8]
  884.     pavgb     m2, [r0+%3+r5]
  885.     mova      m1, [r0+%3+9]
  886.     mova      m0, [r0+%3+1]
  887.     pavgb     m1, [r0+%3+r5+9]
  888.     pavgb     m0, [r0+%3+r5+1]
  889.     pavgb     m1, m3
  890.     pavgb     m0, m2
  891.     mova      m3, m1
  892.     mova      m2, m0
  893.     pand      m1, m7
  894.     pand      m0, m7
  895.     psrlw     m3, 8
  896.     psrlw     m2, 8
  897.     packuswb  m0, m1
  898.     packuswb  m2, m3
  899.     mova    [%1], m0
  900.     mova    [%2], m2
  901. %endmacro
  902. ;-----------------------------------------------------------------------------
  903. ; void frame_init_lowres_core( uint8_t *src0, uint8_t *dst0, uint8_t *dsth, uint8_t *dstv, uint8_t *dstc,
  904. ;                              int src_stride, int dst_stride, int width, int height )
  905. ;-----------------------------------------------------------------------------
  906. %macro FRAME_INIT_LOWRES 1-2 0 ; FIXME
  907. cglobal x264_frame_init_lowres_core_%1, 6,7,%2
  908. %ifdef WIN64
  909.     movsxd   r5, r5d
  910. %endif
  911.     ; src += 2*(height-1)*stride + 2*width
  912.     mov      r6d, r8m
  913.     dec      r6d
  914.     imul     r6d, r5d
  915.     add      r6d, r7m
  916.     lea       r0, [r0+r6*2]
  917.     ; dst += (height-1)*stride + width
  918.     mov      r6d, r8m
  919.     dec      r6d
  920.     imul     r6d, r6m
  921.     add      r6d, r7m
  922.     add       r1, r6
  923.     add       r2, r6
  924.     add       r3, r6
  925.     add       r4, r6
  926.     ; gap = stride - width
  927.     mov      r6d, r6m
  928.     sub      r6d, r7m
  929.     PUSH      r6
  930.     %define dst_gap [rsp+gprsize]
  931.     mov      r6d, r5d
  932.     sub      r6d, r7m
  933.     shl      r6d, 1
  934.     PUSH      r6
  935.     %define src_gap [rsp]
  936. %if mmsize == 16
  937.     ; adjust for the odd end case
  938.     mov      r6d, r7m
  939.     and      r6d, 8
  940.     sub       r1, r6
  941.     sub       r2, r6
  942.     sub       r3, r6
  943.     sub       r4, r6
  944.     add  dst_gap, r6d
  945. %endif ; mmsize
  946.     pcmpeqb   m7, m7
  947.     psrlw     m7, 8
  948. .vloop:
  949.     mov      r6d, r7m
  950. %ifnidn %1, mmxext
  951.     mova      m0, [r0]
  952.     mova      m1, [r0+r5]
  953.     pavgb     m0, m1
  954.     pavgb     m1, [r0+r5*2]
  955. %endif
  956. %if mmsize == 16
  957.     test     r6d, 8
  958.     jz .hloop
  959.     sub       r0, 16
  960.     FILT8x4   m0, m1, m2, m3, m4, m5, 0
  961.     packuswb  m0, m4
  962.     packuswb  m1, m5
  963.     movq    [r1], m0
  964.     movhps  [r2], m0
  965.     movq    [r3], m1
  966.     movhps  [r4], m1
  967.     mova      m0, m2
  968.     mova      m1, m3
  969.     sub      r6d, 8
  970. %endif ; mmsize
  971. .hloop:
  972.     sub       r0, mmsize*2
  973.     sub       r1, mmsize
  974.     sub       r2, mmsize
  975.     sub       r3, mmsize
  976.     sub       r4, mmsize
  977. %ifdef m8
  978.     FILT8x4   m0, m1, m2, m3, m10, m11, mmsize
  979.     mova      m8, m0
  980.     mova      m9, m1
  981.     FILT8x4   m2, m3, m0, m1, m4, m5, 0
  982.     packuswb  m2, m8
  983.     packuswb  m3, m9
  984.     packuswb  m4, m10
  985.     packuswb  m5, m11
  986.     mova    [r1], m2
  987.     mova    [r2], m4
  988.     mova    [r3], m3
  989.     mova    [r4], m5
  990. %elifidn %1, mmxext
  991.     FILT8x2U  r1, r2, 0
  992.     FILT8x2U  r3, r4, r5
  993. %else
  994.     FILT16x2  m0, r1, r2, 0
  995.     FILT16x2  m1, r3, r4, r5
  996. %endif
  997.     sub      r6d, mmsize
  998.     jg .hloop
  999. .skip:
  1000.     mov       r6, dst_gap
  1001.     sub       r0, src_gap
  1002.     sub       r1, r6
  1003.     sub       r2, r6
  1004.     sub       r3, r6
  1005.     sub       r4, r6
  1006.     dec    dword r8m
  1007.     jg .vloop
  1008.     ADD      rsp, 2*gprsize
  1009.     emms
  1010.     RET
  1011. %endmacro ; FRAME_INIT_LOWRES
  1012. INIT_MMX
  1013. %define PALIGNR PALIGNR_MMX
  1014. FRAME_INIT_LOWRES mmxext
  1015. %ifndef ARCH_X86_64
  1016. FRAME_INIT_LOWRES cache32_mmxext
  1017. %endif
  1018. INIT_XMM
  1019. FRAME_INIT_LOWRES sse2, 12
  1020. %define PALIGNR PALIGNR_SSSE3
  1021. FRAME_INIT_LOWRES ssse3, 12
  1022. ;-----------------------------------------------------------------------------
  1023. ; void mbtree_propagate_cost( int *dst, uint16_t *propagate_in, uint16_t *intra_costs,
  1024. ;                             uint16_t *inter_costs, uint16_t *inv_qscales, int len )
  1025. ;-----------------------------------------------------------------------------
  1026. cglobal x264_mbtree_propagate_cost_sse2, 6,6
  1027.     shl r5d, 1
  1028.     lea r0, [r0+r5*2]
  1029.     add r1, r5
  1030.     add r2, r5
  1031.     add r3, r5
  1032.     add r4, r5
  1033.     neg r5
  1034.     pxor      xmm5, xmm5
  1035.     movdqa    xmm4, [pd_128 GLOBAL]
  1036. .loop:
  1037.     movq      xmm2, [r2+r5] ; intra
  1038.     movq      xmm0, [r4+r5] ; invq
  1039.     punpcklwd xmm2, xmm5
  1040.     punpcklwd xmm0, xmm5
  1041.     pmaddwd   xmm0, xmm2
  1042.     paddd     xmm0, xmm4
  1043.     psrld     xmm0, 8       ; intra*invq>>8
  1044.     movq      xmm1, [r1+r5] ; prop
  1045.     movq      xmm3, [r3+r5] ; inter
  1046.     punpcklwd xmm1, xmm5
  1047.     punpcklwd xmm3, xmm5
  1048.     paddd     xmm0, xmm1    ; prop + (intra*invq>>8)
  1049.     cvtdq2ps  xmm1, xmm2    ; intra
  1050.     psubd     xmm2, xmm3    ; intra - inter
  1051.     cvtdq2ps  xmm0, xmm0
  1052.     cvtdq2ps  xmm2, xmm2
  1053.     mulps     xmm0, xmm2    ; (prop + (intra*invq>>8)) * (intra - inter)
  1054.     divps     xmm0, xmm1    ; / intra
  1055.     cvttps2dq xmm0, xmm0    ; truncation isn't really desired, but matches the integer implementation
  1056.     movdqa [r0+r5*2], xmm0
  1057.     add r5, 8
  1058.     jl .loop
  1059.     REP_RET