shithub: libvpx

ref: c202100fe408499e8a18a5878af0fcfc74bfc635
dir: /vpx_scale/win32/scaleopt.c/

View raw version
/*
 *  Copyright (c) 2010 The WebM project authors. All Rights Reserved.
 *
 *  Use of this source code is governed by a BSD-style license
 *  that can be found in the LICENSE file in the root of the source
 *  tree. An additional intellectual property rights grant can be found
 *  in the file PATENTS.  All contributing project authors may
 *  be found in the AUTHORS file in the root of the source tree.
 */


/****************************************************************************
*
*   Module Title :     scaleopt.cpp
*
*   Description  :     Optimized scaling functions
*
****************************************************************************/
#include "pragmas.h"



/****************************************************************************
*  Module Statics
****************************************************************************/
__declspec(align(16)) const static unsigned short one_fifth[]  = { 51, 51, 51, 51 };
__declspec(align(16)) const static unsigned short two_fifths[] = { 102, 102, 102, 102 };
__declspec(align(16)) const static unsigned short three_fifths[] = { 154, 154, 154, 154 };
__declspec(align(16)) const static unsigned short four_fifths[] = { 205, 205, 205, 205 };
__declspec(align(16)) const static unsigned short round_values[] = { 128, 128, 128, 128 };
__declspec(align(16)) const static unsigned short four_ones[] = { 1, 1, 1, 1};
__declspec(align(16)) const static unsigned short const45_2[] = {205, 154, 102,  51 };
__declspec(align(16)) const static unsigned short const45_1[] = { 51, 102, 154, 205 };
__declspec(align(16)) const static unsigned char  mask45[] = { 0, 0, 0, 0, 0, 0, 255, 0};
__declspec(align(16)) const static unsigned short const35_2[] = { 154,  51, 205, 102 };
__declspec(align(16)) const static unsigned short const35_1[] = { 102, 205,  51, 154 };



#include "vpx_scale/vpxscale.h"
#include "vpx_mem/vpx_mem.h"

/****************************************************************************
 *
 *  ROUTINE       : horizontal_line_3_5_scale_mmx
 *
 *  INPUTS        : const unsigned char *source :
 *                  unsigned int source_width    :
 *                  unsigned char *dest         :
 *                  unsigned int dest_width      :
 *
 *  OUTPUTS       : None.
 *
 *  RETURNS       : void
 *
 *  FUNCTION      : 3 to 5 up-scaling of a horizontal line of pixels.
 *
 *  SPECIAL NOTES : None.
 *
 ****************************************************************************/
static
void horizontal_line_3_5_scale_mmx
(
  const unsigned char *source,
  unsigned int source_width,
  unsigned char *dest,
  unsigned int dest_width
) {
  (void) dest_width;

  __asm {

    push ebx

    mov         esi,    source
    mov         edi,    dest

    mov         ecx,    source_width
    lea         edx,    [esi+ecx-3];

    movq        mm5,    const35_1       // mm5 = 66 xx cd xx 33 xx 9a xx
    movq        mm6,    const35_2       // mm6 = 9a xx 33 xx cd xx 66 xx

    movq        mm4,    round_values     // mm4 = 80 xx 80 xx 80 xx 80 xx
    pxor        mm7,    mm7             // clear mm7

    horiz_line_3_5_loop:

    mov        eax,    DWORD PTR [esi] // eax = 00 01 02 03
    mov        ebx,    eax

    and         ebx,    0xffff00        // ebx = xx 01 02 xx
    mov         ecx,    eax             // ecx = 00 01 02 03

    and         eax,    0xffff0000      // eax = xx xx 02 03
    xor         ecx,    eax             // ecx = 00 01 xx xx

    shr         ebx,    8               // ebx = 01 02 xx xx
    or          eax,    ebx             // eax = 01 02 02 03

    shl         ebx,    16              // ebx = xx xx 01 02
    movd        mm1,    eax             // mm1 = 01 02 02 03 xx xx xx xx

    or          ebx,    ecx             // ebx = 00 01 01 02
    punpcklbw   mm1,    mm7             // mm1 = 01 xx 02 xx 02 xx 03 xx

    movd        mm0,    ebx             // mm0 = 00 01 01 02
    pmullw      mm1,    mm6             //

    punpcklbw   mm0,    mm7             // mm0 = 00 xx 01 xx 01 xx 02 xx
    pmullw      mm0,    mm5             //

    mov         [edi],  ebx             // writeoutput 00 xx xx xx
    add         esi,    3

    add         edi,    5
    paddw       mm0,    mm1

    paddw       mm0,    mm4
    psrlw       mm0,    8

    cmp         esi,    edx
    packuswb    mm0,    mm7

    movd        DWORD Ptr [edi-4], mm0
    jl          horiz_line_3_5_loop

// Exit:
    mov         eax,    DWORD PTR [esi] // eax = 00 01 02 03
    mov         ebx,    eax

    and         ebx,    0xffff00        // ebx = xx 01 02 xx
    mov         ecx,    eax             // ecx = 00 01 02 03

    and         eax,    0xffff0000      // eax = xx xx 02 03
    xor         ecx,    eax             // ecx = 00 01 xx xx

    shr         ebx,    8               // ebx = 01 02 xx xx
    or          eax,    ebx             // eax = 01 02 02 03

    shl         eax,    8               // eax = xx 01 02 02
    and         eax,    0xffff0000      // eax = xx xx 02 02

    or          eax,    ebx             // eax = 01 02 02 02

    shl         ebx,    16              // ebx = xx xx 01 02
    movd        mm1,    eax             // mm1 = 01 02 02 02 xx xx xx xx

    or          ebx,    ecx             // ebx = 00 01 01 02
    punpcklbw   mm1,    mm7             // mm1 = 01 xx 02 xx 02 xx 02 xx

    movd        mm0,    ebx             // mm0 = 00 01 01 02
    pmullw      mm1,    mm6             //

    punpcklbw   mm0,    mm7             // mm0 = 00 xx 01 xx 01 xx 02 xx
    pmullw      mm0,    mm5             //

    mov         [edi],  ebx             // writeoutput 00 xx xx xx
    paddw       mm0,    mm1

    paddw       mm0,    mm4
    psrlw       mm0,    8

    packuswb    mm0,    mm7
    movd        DWORD Ptr [edi+1], mm0

    pop ebx

  }

}


/****************************************************************************
 *
 *  ROUTINE       : horizontal_line_4_5_scale_mmx
 *
 *  INPUTS        : const unsigned char *source :
 *                  unsigned int source_width    :
 *                  unsigned char *dest         :
 *                  unsigned int dest_width      :
 *
 *  OUTPUTS       : None.
 *
 *  RETURNS       : void
 *
 *  FUNCTION      : 4 to 5 up-scaling of a horizontal line of pixels.
 *
 *  SPECIAL NOTES : None.
 *
 ****************************************************************************/
static
void horizontal_line_4_5_scale_mmx
(
  const unsigned char *source,
  unsigned int source_width,
  unsigned char *dest,
  unsigned int dest_width
) {
  (void)dest_width;

  __asm {

    mov         esi,    source
    mov         edi,    dest

    mov         ecx,    source_width
    lea         edx,    [esi+ecx-8];

    movq        mm5,    const45_1       // mm5 = 33 xx 66 xx 9a xx cd xx
    movq        mm6,    const45_2       // mm6 = cd xx 9a xx 66 xx 33 xx

    movq        mm4,    round_values     // mm4 = 80 xx 80 xx 80 xx 80 xx
    pxor        mm7,    mm7             // clear mm7

    horiz_line_4_5_loop:

    movq        mm0,    QWORD PTR [esi]           // mm0 = 00 01 02 03 04 05 06 07
    movq        mm1,    QWORD PTR [esi+1];        // mm1 = 01 02 03 04 05 06 07 08

    movq        mm2,    mm0             // mm2 = 00 01 02 03 04 05 06 07
    movq        mm3,    mm1             // mm3 = 01 02 03 04 05 06 07 08

    movd        DWORD PTR [edi],  mm0             // write output 00 xx xx xx
    punpcklbw   mm0,    mm7             // mm0 = 00 xx 01 xx 02 xx 03 xx

    punpcklbw   mm1,    mm7             // mm1 = 01 xx 02 xx 03 xx 04 xx
    pmullw      mm0,    mm5             // 00* 51 01*102 02*154 03*205

    pmullw      mm1,    mm6             // 01*205 02*154 03*102 04* 51
    punpckhbw   mm2,    mm7             // mm2 = 04 xx 05 xx 06 xx 07 xx

    movd        DWORD PTR [edi+5], mm2            // write ouput 05 xx xx xx
    pmullw      mm2,    mm5             // 04* 51 05*102 06*154 07*205

    punpckhbw   mm3,    mm7             // mm3 = 05 xx 06 xx 07 xx 08 xx
    pmullw      mm3,    mm6             // 05*205 06*154 07*102 08* 51

    paddw       mm0,    mm1             // added round values
    paddw       mm0,    mm4

    psrlw       mm0,    8               // output: 01 xx 02 xx 03 xx 04 xx
    packuswb    mm0,    mm7

    movd        DWORD PTR [edi+1], mm0  // write output 01 02 03 04
    add         edi,    10

    add         esi,    8
    paddw       mm2,    mm3             //

    paddw       mm2,    mm4             // added round values
    cmp         esi,    edx

    psrlw       mm2,    8
    packuswb    mm2,    mm7

    movd        DWORD PTR [edi-4], mm2 // writeoutput 06 07 08 09
    jl         horiz_line_4_5_loop

// Exit:
    movq        mm0,    [esi]           // mm0 = 00 01 02 03 04 05 06 07
    movq        mm1,    mm0             // mm1 = 00 01 02 03 04 05 06 07

    movq        mm2,    mm0             // mm2 = 00 01 02 03 04 05 06 07
    psrlq       mm1,    8               // mm1 = 01 02 03 04 05 06 07 00

    movq        mm3,    mask45          // mm3 = 00 00 00 00 00 00 ff 00
    pand        mm3,    mm1             // mm3 = 00 00 00 00 00 00 07 00

    psllq       mm3,    8               // mm3 = 00 00 00 00 00 00 00 07
    por         mm1,    mm3             // mm1 = 01 02 03 04 05 06 07 07

    movq        mm3,    mm1

    movd        DWORD PTR [edi],  mm0   // write output 00 xx xx xx
    punpcklbw   mm0,    mm7             // mm0 = 00 xx 01 xx 02 xx 03 xx

    punpcklbw   mm1,    mm7             // mm1 = 01 xx 02 xx 03 xx 04 xx
    pmullw      mm0,    mm5             // 00* 51 01*102 02*154 03*205

    pmullw      mm1,    mm6             // 01*205 02*154 03*102 04* 51
    punpckhbw   mm2,    mm7             // mm2 = 04 xx 05 xx 06 xx 07 xx

    movd        DWORD PTR [edi+5], mm2  // write ouput 05 xx xx xx
    pmullw      mm2,    mm5             // 04* 51 05*102 06*154 07*205

    punpckhbw   mm3,    mm7             // mm3 = 05 xx 06 xx 07 xx 08 xx
    pmullw      mm3,    mm6             // 05*205 06*154 07*102 07* 51

    paddw       mm0,    mm1             // added round values
    paddw       mm0,    mm4

    psrlw       mm0,    8               // output: 01 xx 02 xx 03 xx 04 xx
    packuswb    mm0,    mm7             // 01 02 03 04 xx xx xx xx

    movd        DWORD PTR [edi+1], mm0  // write output 01 02 03 04
    paddw       mm2,    mm3             //

    paddw       mm2,    mm4             // added round values
    psrlw       mm2,    8

    packuswb    mm2,    mm7
    movd        DWORD PTR [edi+6], mm2  // writeoutput 06 07 08 09


  }
}

/****************************************************************************
 *
 *  ROUTINE       : vertical_band_4_5_scale_mmx
 *
 *  INPUTS        : unsigned char *dest    :
 *                  unsigned int dest_pitch :
 *                  unsigned int dest_width :
 *
 *  OUTPUTS       : None.
 *
 *  RETURNS       : void
 *
 *  FUNCTION      : 4 to 5 up-scaling of a 4 pixel high band of pixels.
 *
 *  SPECIAL NOTES : The routine uses the first line of the band below
 *                  the current band. The function also has a "C" only
 *                  version.
 *
 ****************************************************************************/
static
void vertical_band_4_5_scale_mmx
(
  unsigned char *dest,
  unsigned int dest_pitch,
  unsigned int dest_width
) {
  __asm {

    mov         esi,    dest                    // Get the source and destination pointer
    mov         ecx,    dest_pitch               // Get the pitch size

    lea         edi,    [esi+ecx*2]             // tow lines below
    add         edi,    ecx                     // three lines below

    pxor        mm7,    mm7                     // clear out mm7
    mov         edx,    dest_width               // Loop counter

    vs_4_5_loop:

    movq        mm0,    QWORD ptr [esi]         // src[0];
    movq        mm1,    QWORD ptr [esi+ecx]     // src[1];

    movq        mm2,    mm0                     // Make a copy
    punpcklbw   mm0,    mm7                     // unpack low to word

    movq        mm5,    one_fifth
    punpckhbw   mm2,    mm7                     // unpack high to word

    pmullw      mm0,    mm5                     // a * 1/5

    movq        mm3,    mm1                     // make a copy
    punpcklbw   mm1,    mm7                     // unpack low to word

    pmullw      mm2,    mm5                     // a * 1/5
    movq        mm6,    four_fifths               // constan

    movq        mm4,    mm1                     // copy of low b
    pmullw      mm4,    mm6                     // b * 4/5

    punpckhbw   mm3,    mm7                     // unpack high to word
    movq        mm5,    mm3                     // copy of high b

    pmullw      mm5,    mm6                     // b * 4/5
    paddw       mm0,    mm4                     // a * 1/5 + b * 4/5

    paddw       mm2,    mm5                     // a * 1/5 + b * 4/5
    paddw       mm0,    round_values             // + 128

    paddw       mm2,    round_values             // + 128
    psrlw       mm0,    8

    psrlw       mm2,    8
    packuswb    mm0,    mm2                     // des [1]

    movq        QWORD ptr [esi+ecx], mm0        // write des[1]
    movq        mm0,    [esi+ecx*2]             // mm0 = src[2]

    // mm1, mm3 --- Src[1]
    // mm0 --- Src[2]
    // mm7 for unpacking

    movq        mm5,    two_fifths
    movq        mm2,    mm0                     // make a copy

    pmullw      mm1,    mm5                     // b * 2/5
    movq        mm6,    three_fifths


    punpcklbw   mm0,    mm7                     // unpack low to word
    pmullw      mm3,    mm5                     // b * 2/5

    movq        mm4,    mm0                     // make copy of c
    punpckhbw   mm2,    mm7                     // unpack high to word

    pmullw      mm4,    mm6                     // c * 3/5
    movq        mm5,    mm2

    pmullw      mm5,    mm6                     // c * 3/5
    paddw       mm1,    mm4                     // b * 2/5 + c * 3/5

    paddw       mm3,    mm5                     // b * 2/5 + c * 3/5
    paddw       mm1,    round_values             // + 128

    paddw       mm3,    round_values             // + 128
    psrlw       mm1,    8

    psrlw       mm3,    8
    packuswb    mm1,    mm3                     // des[2]

    movq        QWORD ptr [esi+ecx*2], mm1      // write des[2]
    movq        mm1,    [edi]                   // mm1=Src[3];

    // mm0, mm2 --- Src[2]
    // mm1 --- Src[3]
    // mm6 --- 3/5
    // mm7 for unpacking

    pmullw      mm0,    mm6                     // c * 3/5
    movq        mm5,    two_fifths               // mm5 = 2/5

    movq        mm3,    mm1                     // make a copy
    pmullw      mm2,    mm6                     // c * 3/5

    punpcklbw   mm1,    mm7                     // unpack low
    movq        mm4,    mm1                     // make a copy

    punpckhbw   mm3,    mm7                     // unpack high
    pmullw      mm4,    mm5                     // d * 2/5

    movq        mm6,    mm3                     // make a copy
    pmullw      mm6,    mm5                     // d * 2/5

    paddw       mm0,    mm4                     // c * 3/5 + d * 2/5
    paddw       mm2,    mm6                     // c * 3/5 + d * 2/5

    paddw       mm0,    round_values             // + 128
    paddw       mm2,    round_values             // + 128

    psrlw       mm0,    8
    psrlw       mm2,    8

    packuswb    mm0,    mm2                     // des[3]
    movq        QWORD ptr [edi], mm0            // write des[3]

    //  mm1, mm3 --- Src[3]
    //  mm7 -- cleared for unpacking

    movq        mm0,    [edi+ecx*2]             // mm0, Src[0] of the next group

    movq        mm5,    four_fifths              // mm5 = 4/5
    pmullw      mm1,    mm5                     // d * 4/5

    movq        mm6,    one_fifth                // mm6 = 1/5
    movq        mm2,    mm0                     // make a copy

    pmullw      mm3,    mm5                     // d * 4/5
    punpcklbw   mm0,    mm7                     // unpack low

    pmullw      mm0,    mm6                     // an * 1/5
    punpckhbw   mm2,    mm7                     // unpack high

    paddw       mm1,    mm0                     // d * 4/5 + an * 1/5
    pmullw      mm2,    mm6                     // an * 1/5

    paddw       mm3,    mm2                     // d * 4/5 + an * 1/5
    paddw       mm1,    round_values             // + 128

    paddw       mm3,    round_values             // + 128
    psrlw       mm1,    8

    psrlw       mm3,    8
    packuswb    mm1,    mm3                     // des[4]

    movq        QWORD ptr [edi+ecx], mm1        // write des[4]

    add         edi,    8
    add         esi,    8

    sub         edx,    8
    jg         vs_4_5_loop
  }
}

/****************************************************************************
 *
 *  ROUTINE       : last_vertical_band_4_5_scale_mmx
 *
 *  INPUTS        : unsigned char *dest    :
 *                  unsigned int dest_pitch :
 *                  unsigned int dest_width :
 *
 *  OUTPUTS       : None.
 *
 *  RETURNS       : None
 *
 *  FUNCTION      : 4 to 5 up-scaling of the last 4-pixel high band in an image.
 *
 *  SPECIAL NOTES : The routine uses the first line of the band below
 *                  the current band. The function also has an "C" only
 *                  version.
 *
 ****************************************************************************/
static
void last_vertical_band_4_5_scale_mmx
(
  unsigned char *dest,
  unsigned int dest_pitch,
  unsigned int dest_width
) {
  __asm {
    mov         esi,    dest                    // Get the source and destination pointer
    mov         ecx,    dest_pitch               // Get the pitch size

    lea         edi,    [esi+ecx*2]             // tow lines below
    add         edi,    ecx                     // three lines below

    pxor        mm7,    mm7                     // clear out mm7
    mov         edx,    dest_width               // Loop counter

    last_vs_4_5_loop:

    movq        mm0,    QWORD ptr [esi]         // src[0];
    movq        mm1,    QWORD ptr [esi+ecx]     // src[1];

    movq        mm2,    mm0                     // Make a copy
    punpcklbw   mm0,    mm7                     // unpack low to word

    movq        mm5,    one_fifth
    punpckhbw   mm2,    mm7                     // unpack high to word

    pmullw      mm0,    mm5                     // a * 1/5

    movq        mm3,    mm1                     // make a copy
    punpcklbw   mm1,    mm7                     // unpack low to word

    pmullw      mm2,    mm5                     // a * 1/5
    movq        mm6,    four_fifths               // constan

    movq        mm4,    mm1                     // copy of low b
    pmullw      mm4,    mm6                     // b * 4/5

    punpckhbw   mm3,    mm7                     // unpack high to word
    movq        mm5,    mm3                     // copy of high b

    pmullw      mm5,    mm6                     // b * 4/5
    paddw       mm0,    mm4                     // a * 1/5 + b * 4/5

    paddw       mm2,    mm5                     // a * 1/5 + b * 4/5
    paddw       mm0,    round_values             // + 128

    paddw       mm2,    round_values             // + 128
    psrlw       mm0,    8

    psrlw       mm2,    8
    packuswb    mm0,    mm2                     // des [1]

    movq        QWORD ptr [esi+ecx], mm0        // write des[1]
    movq        mm0,    [esi+ecx*2]             // mm0 = src[2]

    // mm1, mm3 --- Src[1]
    // mm0 --- Src[2]
    // mm7 for unpacking

    movq        mm5,    two_fifths
    movq        mm2,    mm0                     // make a copy

    pmullw      mm1,    mm5                     // b * 2/5
    movq        mm6,    three_fifths


    punpcklbw   mm0,    mm7                     // unpack low to word
    pmullw      mm3,    mm5                     // b * 2/5

    movq        mm4,    mm0                     // make copy of c
    punpckhbw   mm2,    mm7                     // unpack high to word

    pmullw      mm4,    mm6                     // c * 3/5
    movq        mm5,    mm2

    pmullw      mm5,    mm6                     // c * 3/5
    paddw       mm1,    mm4                     // b * 2/5 + c * 3/5

    paddw       mm3,    mm5                     // b * 2/5 + c * 3/5
    paddw       mm1,    round_values             // + 128

    paddw       mm3,    round_values             // + 128
    psrlw       mm1,    8

    psrlw       mm3,    8
    packuswb    mm1,    mm3                     // des[2]

    movq        QWORD ptr [esi+ecx*2], mm1      // write des[2]
    movq        mm1,    [edi]                   // mm1=Src[3];

    movq        QWORD ptr [edi+ecx], mm1        // write des[4];

    // mm0, mm2 --- Src[2]
    // mm1 --- Src[3]
    // mm6 --- 3/5
    // mm7 for unpacking

    pmullw      mm0,    mm6                     // c * 3/5
    movq        mm5,    two_fifths               // mm5 = 2/5

    movq        mm3,    mm1                     // make a copy
    pmullw      mm2,    mm6                     // c * 3/5

    punpcklbw   mm1,    mm7                     // unpack low
    movq        mm4,    mm1                     // make a copy

    punpckhbw   mm3,    mm7                     // unpack high
    pmullw      mm4,    mm5                     // d * 2/5

    movq        mm6,    mm3                     // make a copy
    pmullw      mm6,    mm5                     // d * 2/5

    paddw       mm0,    mm4                     // c * 3/5 + d * 2/5
    paddw       mm2,    mm6                     // c * 3/5 + d * 2/5

    paddw       mm0,    round_values             // + 128
    paddw       mm2,    round_values             // + 128

    psrlw       mm0,    8
    psrlw       mm2,    8

    packuswb    mm0,    mm2                     // des[3]
    movq        QWORD ptr [edi], mm0            // write des[3]

    //  mm1, mm3 --- Src[3]
    //  mm7 -- cleared for unpacking
    add         edi,    8
    add         esi,    8

    sub         edx,    8
    jg          last_vs_4_5_loop
  }
}

/****************************************************************************
 *
 *  ROUTINE       : vertical_band_3_5_scale_mmx
 *
 *  INPUTS        : unsigned char *dest    :
 *                  unsigned int dest_pitch :
 *                  unsigned int dest_width :
 *
 *  OUTPUTS       : None.
 *
 *  RETURNS       : void
 *
 *  FUNCTION      : 3 to 5 up-scaling of a 3-pixel high band of pixels.
 *
 *  SPECIAL NOTES : The routine uses the first line of the band below
 *                  the current band. The function also has an "C" only
 *                  version.
 *
 ****************************************************************************/
static
void vertical_band_3_5_scale_mmx
(
  unsigned char *dest,
  unsigned int dest_pitch,
  unsigned int dest_width
) {
  __asm {
    mov         esi,    dest                    // Get the source and destination pointer
    mov         ecx,    dest_pitch               // Get the pitch size

    lea         edi,    [esi+ecx*2]             // tow lines below
    add         edi,    ecx                     // three lines below

    pxor        mm7,    mm7                     // clear out mm7
    mov         edx,    dest_width               // Loop counter

    vs_3_5_loop:

    movq        mm0,    QWORD ptr [esi]         // src[0];
    movq        mm1,    QWORD ptr [esi+ecx]     // src[1];

    movq        mm2,    mm0                     // Make a copy
    punpcklbw   mm0,    mm7                     // unpack low to word

    movq        mm5,    two_fifths               // mm5 = 2/5
    punpckhbw   mm2,    mm7                     // unpack high to word

    pmullw      mm0,    mm5                     // a * 2/5

    movq        mm3,    mm1                     // make a copy
    punpcklbw   mm1,    mm7                     // unpack low to word

    pmullw      mm2,    mm5                     // a * 2/5
    movq        mm6,    three_fifths             // mm6 = 3/5

    movq        mm4,    mm1                     // copy of low b
    pmullw      mm4,    mm6                     // b * 3/5

    punpckhbw   mm3,    mm7                     // unpack high to word
    movq        mm5,    mm3                     // copy of high b

    pmullw      mm5,    mm6                     // b * 3/5
    paddw       mm0,    mm4                     // a * 2/5 + b * 3/5

    paddw       mm2,    mm5                     // a * 2/5 + b * 3/5
    paddw       mm0,    round_values             // + 128

    paddw       mm2,    round_values             // + 128
    psrlw       mm0,    8

    psrlw       mm2,    8
    packuswb    mm0,    mm2                     // des [1]

    movq        QWORD ptr [esi+ecx], mm0        // write des[1]
    movq        mm0,    [esi+ecx*2]             // mm0 = src[2]

    // mm1, mm3 --- Src[1]
    // mm0 --- Src[2]
    // mm7 for unpacking

    movq        mm4,    mm1                     // b low
    pmullw      mm1,    four_fifths              // b * 4/5 low

    movq        mm5,    mm3                     // b high
    pmullw      mm3,    four_fifths              // b * 4/5 high

    movq        mm2,    mm0                     // c
    pmullw      mm4,    one_fifth                // b * 1/5

    punpcklbw   mm0,    mm7                     // c low
    pmullw      mm5,    one_fifth                // b * 1/5

    movq        mm6,    mm0                     // make copy of c low
    punpckhbw   mm2,    mm7                     // c high

    pmullw      mm6,    one_fifth                // c * 1/5 low
    movq        mm7,    mm2                     // make copy of c high

    pmullw      mm7,    one_fifth                // c * 1/5 high
    paddw       mm1,    mm6                     // b * 4/5 + c * 1/5 low

    paddw       mm3,    mm7                     // b * 4/5 + c * 1/5 high
    movq        mm6,    mm0                     // make copy of c low

    pmullw      mm6,    four_fifths              // c * 4/5 low
    movq        mm7,    mm2                     // make copy of c high

    pmullw      mm7,    four_fifths              // c * 4/5 high

    paddw       mm4,    mm6                     // b * 1/5 + c * 4/5 low
    paddw       mm5,    mm7                     // b * 1/5 + c * 4/5 high

    paddw       mm1,    round_values             // + 128
    paddw       mm3,    round_values             // + 128

    psrlw       mm1,    8
    psrlw       mm3,    8

    packuswb    mm1,    mm3                     // des[2]
    movq        QWORD ptr [esi+ecx*2], mm1      // write des[2]

    paddw       mm4,    round_values             // + 128
    paddw       mm5,    round_values             // + 128

    psrlw       mm4,    8
    psrlw       mm5,    8

    packuswb    mm4,    mm5                     // des[3]
    movq        QWORD ptr [edi], mm4            // write des[3]

    //  mm0, mm2 --- Src[3]

    pxor        mm7,    mm7                     // clear mm7 for unpacking
    movq        mm1,    [edi+ecx*2]             // mm1 = Src[0] of the next group

    movq        mm5,    three_fifths             // mm5 = 3/5
    pmullw      mm0,    mm5                     // d * 3/5

    movq        mm6,    two_fifths                // mm6 = 2/5
    movq        mm3,    mm1                     // make a copy

    pmullw      mm2,    mm5                     // d * 3/5
    punpcklbw   mm1,    mm7                     // unpack low

    pmullw      mm1,    mm6                     // an * 2/5
    punpckhbw   mm3,    mm7                     // unpack high

    paddw       mm0,    mm1                     // d * 3/5 + an * 2/5
    pmullw      mm3,    mm6                     // an * 2/5

    paddw       mm2,    mm3                     // d * 3/5 + an * 2/5
    paddw       mm0,    round_values             // + 128

    paddw       mm2,    round_values             // + 128
    psrlw       mm0,    8

    psrlw       mm2,    8
    packuswb    mm0,    mm2                     // des[4]

    movq        QWORD ptr [edi+ecx], mm0        // write des[4]

    add         edi,    8
    add         esi,    8

    sub         edx,    8
    jg          vs_3_5_loop
  }
}

/****************************************************************************
 *
 *  ROUTINE       : last_vertical_band_3_5_scale_mmx
 *
 *  INPUTS        : unsigned char *dest    :
 *                  unsigned int dest_pitch :
 *                  unsigned int dest_width :
 *
 *  OUTPUTS       : None.
 *
 *  RETURNS       : void
 *
 *  FUNCTION      : 3 to 5 up-scaling of a 3-pixel high band of pixels.
 *
 *  SPECIAL NOTES : The routine uses the first line of the band below
 *                  the current band. The function also has an "C" only
 *                  version.
 *
 ****************************************************************************/
static
void last_vertical_band_3_5_scale_mmx
(
  unsigned char *dest,
  unsigned int dest_pitch,
  unsigned int dest_width
) {
  __asm {
    mov         esi,    dest                    // Get the source and destination pointer
    mov         ecx,    dest_pitch               // Get the pitch size

    lea         edi,    [esi+ecx*2]             // tow lines below
    add         edi,    ecx                     // three lines below

    pxor        mm7,    mm7                     // clear out mm7
    mov         edx,    dest_width               // Loop counter


    last_vs_3_5_loop:

    movq        mm0,    QWORD ptr [esi]         // src[0];
    movq        mm1,    QWORD ptr [esi+ecx]     // src[1];

    movq        mm2,    mm0                     // Make a copy
    punpcklbw   mm0,    mm7                     // unpack low to word

    movq        mm5,    two_fifths               // mm5 = 2/5
    punpckhbw   mm2,    mm7                     // unpack high to word

    pmullw      mm0,    mm5                     // a * 2/5

    movq        mm3,    mm1                     // make a copy
    punpcklbw   mm1,    mm7                     // unpack low to word

    pmullw      mm2,    mm5                     // a * 2/5
    movq        mm6,    three_fifths             // mm6 = 3/5

    movq        mm4,    mm1                     // copy of low b
    pmullw      mm4,    mm6                     // b * 3/5

    punpckhbw   mm3,    mm7                     // unpack high to word
    movq        mm5,    mm3                     // copy of high b

    pmullw      mm5,    mm6                     // b * 3/5
    paddw       mm0,    mm4                     // a * 2/5 + b * 3/5

    paddw       mm2,    mm5                     // a * 2/5 + b * 3/5
    paddw       mm0,    round_values             // + 128

    paddw       mm2,    round_values             // + 128
    psrlw       mm0,    8

    psrlw       mm2,    8
    packuswb    mm0,    mm2                     // des [1]

    movq        QWORD ptr [esi+ecx], mm0        // write des[1]
    movq        mm0,    [esi+ecx*2]             // mm0 = src[2]



    // mm1, mm3 --- Src[1]
    // mm0 --- Src[2]
    // mm7 for unpacking

    movq        mm4,    mm1                     // b low
    pmullw      mm1,    four_fifths              // b * 4/5 low

    movq        QWORD ptr [edi+ecx], mm0        // write des[4]

    movq        mm5,    mm3                     // b high
    pmullw      mm3,    four_fifths              // b * 4/5 high

    movq        mm2,    mm0                     // c
    pmullw      mm4,    one_fifth                // b * 1/5

    punpcklbw   mm0,    mm7                     // c low
    pmullw      mm5,    one_fifth                // b * 1/5

    movq        mm6,    mm0                     // make copy of c low
    punpckhbw   mm2,    mm7                     // c high

    pmullw      mm6,    one_fifth                // c * 1/5 low
    movq        mm7,    mm2                     // make copy of c high

    pmullw      mm7,    one_fifth                // c * 1/5 high
    paddw       mm1,    mm6                     // b * 4/5 + c * 1/5 low

    paddw       mm3,    mm7                     // b * 4/5 + c * 1/5 high
    movq        mm6,    mm0                     // make copy of c low

    pmullw      mm6,    four_fifths              // c * 4/5 low
    movq        mm7,    mm2                     // make copy of c high

    pmullw      mm7,    four_fifths              // c * 4/5 high

    paddw       mm4,    mm6                     // b * 1/5 + c * 4/5 low
    paddw       mm5,    mm7                     // b * 1/5 + c * 4/5 high

    paddw       mm1,    round_values             // + 128
    paddw       mm3,    round_values             // + 128

    psrlw       mm1,    8
    psrlw       mm3,    8

    packuswb    mm1,    mm3                     // des[2]
    movq        QWORD ptr [esi+ecx*2], mm1      // write des[2]

    paddw       mm4,    round_values             // + 128
    paddw       mm5,    round_values             // + 128

    psrlw       mm4,    8
    psrlw       mm5,    8

    packuswb    mm4,    mm5                     // des[3]
    movq        QWORD ptr [edi], mm4            // write des[3]

    //  mm0, mm2 --- Src[3]

    add         edi,    8
    add         esi,    8

    sub         edx,    8
    jg          last_vs_3_5_loop
  }
}

/****************************************************************************
 *
 *  ROUTINE       : vertical_band_1_2_scale_mmx
 *
 *  INPUTS        : unsigned char *dest    :
 *                  unsigned int dest_pitch :
 *                  unsigned int dest_width :
 *
 *  OUTPUTS       : None.
 *
 *  RETURNS       : void
 *
 *  FUNCTION      : 1 to 2 up-scaling of a band of pixels.
 *
 *  SPECIAL NOTES : The routine uses the first line of the band below
 *                  the current band. The function also has an "C" only
 *                  version.
 *
 ****************************************************************************/
static
void vertical_band_1_2_scale_mmx
(
  unsigned char *dest,
  unsigned int dest_pitch,
  unsigned int dest_width
) {
  __asm {

    mov         esi,    dest                    // Get the source and destination pointer
    mov         ecx,    dest_pitch               // Get the pitch size

    pxor        mm7,    mm7                     // clear out mm7
    mov         edx,    dest_width               // Loop counter

    vs_1_2_loop:

    movq        mm0,    [esi]                   // get Src[0]
    movq        mm1,    [esi + ecx * 2]         // get Src[1]

    movq        mm2,    mm0                     // make copy before unpack
    movq        mm3,    mm1                     // make copy before unpack

    punpcklbw   mm0,    mm7                     // low Src[0]
    movq        mm6,    four_ones                // mm6= 1, 1, 1, 1

    punpcklbw   mm1,    mm7                     // low Src[1]
    paddw       mm0,    mm1                     // low (a + b)

    punpckhbw   mm2,    mm7                     // high Src[0]
    paddw       mm0,    mm6                     // low (a + b + 1)

    punpckhbw   mm3,    mm7
    paddw       mm2,    mm3                     // high (a + b )

    psraw       mm0,    1                       // low (a + b +1 )/2
    paddw       mm2,    mm6                     // high (a + b + 1)

    psraw       mm2,    1                       // high (a + b + 1)/2
    packuswb    mm0,    mm2                     // pack results

    movq        [esi+ecx], mm0                  // write out eight bytes
    add         esi,    8

    sub         edx,    8
    jg          vs_1_2_loop
  }

}

/****************************************************************************
 *
 *  ROUTINE       : last_vertical_band_1_2_scale_mmx
 *
 *  INPUTS        : unsigned char *dest    :
 *                  unsigned int dest_pitch :
 *                  unsigned int dest_width :
 *
 *  OUTPUTS       : None.
 *
 *  RETURNS       : void
 *
 *  FUNCTION      : 1 to 2 up-scaling of band of pixels.
 *
 *  SPECIAL NOTES : The routine uses the first line of the band below
 *                  the current band. The function also has an "C" only
 *                  version.
 *
 ****************************************************************************/
static
void last_vertical_band_1_2_scale_mmx
(
  unsigned char *dest,
  unsigned int dest_pitch,
  unsigned int dest_width
) {
  __asm {
    mov         esi,    dest                    // Get the source and destination pointer
    mov         ecx,    dest_pitch               // Get the pitch size

    mov         edx,    dest_width               // Loop counter

    last_vs_1_2_loop:

    movq        mm0,    [esi]                   // get Src[0]
    movq        [esi+ecx], mm0                  // write out eight bytes

    add         esi,    8
    sub         edx,    8

    jg         last_vs_1_2_loop
  }
}

/****************************************************************************
 *
 *  ROUTINE       : horizontal_line_1_2_scale
 *
 *  INPUTS        : const unsigned char *source :
 *                  unsigned int source_width    :
 *                  unsigned char *dest         :
 *                  unsigned int dest_width      :
 *
 *  OUTPUTS       : None.
 *
 *  RETURNS       : void
 *
 *  FUNCTION      : 1 to 2 up-scaling of a horizontal line of pixels.
 *
 *  SPECIAL NOTES : None.
 *
 ****************************************************************************/
static
void horizontal_line_1_2_scale_mmx
(
  const unsigned char *source,
  unsigned int source_width,
  unsigned char *dest,
  unsigned int dest_width
) {
  (void) dest_width;

  __asm {
    mov         esi,    source
    mov         edi,    dest

    pxor        mm7,    mm7
    movq        mm6,    four_ones

    mov         ecx,    source_width

    hs_1_2_loop:

    movq        mm0,    [esi]
    movq        mm1,    [esi+1]

    movq        mm2,    mm0
    movq        mm3,    mm1

    movq        mm4,    mm0
    punpcklbw   mm0,    mm7

    punpcklbw   mm1,    mm7
    paddw       mm0,    mm1

    paddw       mm0,    mm6
    punpckhbw   mm2,    mm7

    punpckhbw   mm3,    mm7
    paddw       mm2,    mm3

    paddw       mm2,    mm6
    psraw       mm0,    1

    psraw       mm2,    1
    packuswb    mm0,    mm2

    movq        mm2,    mm4
    punpcklbw   mm2,    mm0

    movq        [edi],  mm2
    punpckhbw   mm4,    mm0

    movq        [edi+8], mm4
    add         esi,    8

    add         edi,    16
    sub         ecx,    8

    cmp         ecx,    8
    jg          hs_1_2_loop

// last eight pixel

    movq        mm0,    [esi]
    movq        mm1,    mm0

    movq        mm2,    mm0
    movq        mm3,    mm1

    psrlq       mm1,    8
    psrlq       mm3,    56

    psllq       mm3,    56
    por         mm1,    mm3

    movq        mm3,    mm1
    movq        mm4,    mm0

    punpcklbw   mm0,    mm7
    punpcklbw   mm1,    mm7

    paddw       mm0,    mm1
    paddw       mm0,    mm6

    punpckhbw   mm2,    mm7
    punpckhbw   mm3,    mm7

    paddw       mm2,    mm3
    paddw       mm2,    mm6

    psraw       mm0,    1
    psraw       mm2,    1

    packuswb    mm0,    mm2
    movq        mm2,    mm4

    punpcklbw   mm2,    mm0
    movq        [edi],  mm2

    punpckhbw   mm4,    mm0
    movq        [edi+8], mm4
  }
}





__declspec(align(16)) const static unsigned short const54_2[] = {  0,  64, 128, 192 };
__declspec(align(16)) const static unsigned short const54_1[] = {256, 192, 128,  64 };


/****************************************************************************
 *
 *  ROUTINE       : horizontal_line_5_4_scale_mmx
 *
 *  INPUTS        : const unsigned char *source : Pointer to source data.
 *                  unsigned int source_width    : Stride of source.
 *                  unsigned char *dest         : Pointer to destination data.
 *                  unsigned int dest_width      : Stride of destination (NOT USED).
 *
 *  OUTPUTS       : None.
 *
 *  RETURNS       : void
 *
 *  FUNCTION      : Copies horizontal line of pixels from source to
 *                  destination scaling up by 4 to 5.
 *
 *  SPECIAL NOTES : None.
 *
 ****************************************************************************/
static
void horizontal_line_5_4_scale_mmx
(
  const unsigned char *source,
  unsigned int source_width,
  unsigned char *dest,
  unsigned int dest_width
) {
  /*
  unsigned i;
  unsigned int a, b, c, d, e;
  unsigned char *des = dest;
  const unsigned char *src = source;

  (void) dest_width;

  for ( i=0; i<source_width; i+=5 )
  {
      a = src[0];
      b = src[1];
      c = src[2];
      d = src[3];
      e = src[4];

      des[0] = a;
      des[1] = ((b*192 + c* 64 + 128)>>8);
      des[2] = ((c*128 + d*128 + 128)>>8);
      des[3] = ((d* 64 + e*192 + 128)>>8);

      src += 5;
      des += 4;
  }
  */
  (void) dest_width;

  __asm {

    mov         esi,        source;
    mov         edi,        dest;

    mov         ecx,        source_width;
    movq        mm5,        const54_1;

    pxor        mm7,        mm7;
    movq        mm6,        const54_2;

    movq        mm4,        round_values;
    lea         edx,        [esi+ecx];
    horizontal_line_5_4_loop:

    movq        mm0,        QWORD PTR  [esi];
    00 01 02 03 04 05 06 07
    movq        mm1,        mm0;
    00 01 02 03 04 05 06 07

    psrlq       mm0,        8;
    01 02 03 04 05 06 07 xx
    punpcklbw   mm1,        mm7;
    xx 00 xx 01 xx 02 xx 03

    punpcklbw   mm0,        mm7;
    xx 01 xx 02 xx 03 xx 04
    pmullw      mm1,        mm5

    pmullw      mm0,        mm6
    add         esi,        5

    add         edi,        4
    paddw       mm1,        mm0

    paddw       mm1,        mm4
    psrlw       mm1,        8

    cmp         esi,        edx
    packuswb    mm1,        mm7

    movd        DWORD PTR [edi-4], mm1

    jl          horizontal_line_5_4_loop

  }

}
__declspec(align(16)) const static unsigned short one_fourths[]   = {  64,  64,  64, 64  };
__declspec(align(16)) const static unsigned short two_fourths[]   = { 128, 128, 128, 128 };
__declspec(align(16)) const static unsigned short three_fourths[] = { 192, 192, 192, 192 };

static
void vertical_band_5_4_scale_mmx(unsigned char *source, unsigned int src_pitch, unsigned char *dest, unsigned int dest_pitch, unsigned int dest_width) {

  __asm {
    push        ebx

    mov         esi,    source                    // Get the source and destination pointer
    mov         ecx,    src_pitch               // Get the pitch size

    mov         edi,    dest                    // tow lines below
    pxor        mm7,    mm7                     // clear out mm7

    mov         edx,    dest_pitch               // Loop counter
    mov         ebx,    dest_width

    vs_5_4_loop:

    movd        mm0,    DWORD ptr [esi]         // src[0];
    movd        mm1,    DWORD ptr [esi+ecx]     // src[1];

    movd        mm2,    DWORD ptr [esi+ecx*2]
    lea         eax,    [esi+ecx*2]             //

    punpcklbw   mm1,    mm7
    punpcklbw   mm2,    mm7

    movq        mm3,    mm2
    pmullw      mm1,    three_fourths

    pmullw      mm2,    one_fourths
    movd        mm4,    [eax+ecx]

    pmullw      mm3,    two_fourths
    punpcklbw   mm4,    mm7

    movq        mm5,    mm4
    pmullw      mm4,    two_fourths

    paddw       mm1,    mm2
    movd        mm6,    [eax+ecx*2]

    pmullw      mm5,    one_fourths
    paddw       mm1,    round_values;

    paddw       mm3,    mm4
    psrlw       mm1,    8

    punpcklbw   mm6,    mm7
    paddw       mm3,    round_values

    pmullw      mm6,    three_fourths
    psrlw       mm3,    8

    packuswb    mm1,    mm7
    packuswb    mm3,    mm7

    movd        DWORD PTR [edi], mm0
    movd        DWORD PTR [edi+edx], mm1


    paddw       mm5,    mm6
    movd        DWORD PTR [edi+edx*2], mm3

    lea         eax,    [edi+edx*2]
    paddw       mm5,    round_values

    psrlw       mm5,    8
    add         edi,    4

    packuswb    mm5,    mm7
    movd        DWORD PTR [eax+edx], mm5

    add         esi,    4
    sub         ebx,    4

    jg         vs_5_4_loop

    pop         ebx
  }
}


__declspec(align(16)) const static unsigned short const53_1[] = {  0,  85, 171, 0 };
__declspec(align(16)) const static unsigned short const53_2[] = {256, 171,  85, 0 };


static
void horizontal_line_5_3_scale_mmx
(
  const unsigned char *source,
  unsigned int source_width,
  unsigned char *dest,
  unsigned int dest_width
) {

  (void) dest_width;
  __asm {

    mov         esi,        source;
    mov         edi,        dest;

    mov         ecx,        source_width;
    movq        mm5,        const53_1;

    pxor        mm7,        mm7;
    movq        mm6,        const53_2;

    movq        mm4,        round_values;
    lea         edx,        [esi+ecx-5];
    horizontal_line_5_3_loop:

    movq        mm0,        QWORD PTR  [esi];
    00 01 02 03 04 05 06 07
    movq        mm1,        mm0;
    00 01 02 03 04 05 06 07

    psllw       mm0,        8;
    xx 00 xx 02 xx 04 xx 06
    psrlw       mm1,        8;
    01 xx 03 xx 05 xx 07 xx

    psrlw       mm0,        8;
    00 xx 02 xx 04 xx 06 xx
    psllq       mm1,        16;
    xx xx 01 xx 03 xx 05 xx

    pmullw      mm0,        mm6

    pmullw      mm1,        mm5
    add         esi,        5

    add         edi,        3
    paddw       mm1,        mm0

    paddw       mm1,        mm4
    psrlw       mm1,        8

    cmp         esi,        edx
    packuswb    mm1,        mm7

    movd        DWORD PTR [edi-3], mm1
    jl          horizontal_line_5_3_loop

// exit condition
    movq        mm0,        QWORD PTR  [esi];
    00 01 02 03 04 05 06 07
    movq        mm1,        mm0;
    00 01 02 03 04 05 06 07

    psllw       mm0,        8;
    xx 00 xx 02 xx 04 xx 06
    psrlw       mm1,        8;
    01 xx 03 xx 05 xx 07 xx

    psrlw       mm0,        8;
    00 xx 02 xx 04 xx 06 xx
    psllq       mm1,        16;
    xx xx 01 xx 03 xx 05 xx

    pmullw      mm0,        mm6

    pmullw      mm1,        mm5
    paddw       mm1,        mm0

    paddw       mm1,        mm4
    psrlw       mm1,        8

    packuswb    mm1,        mm7
    movd        eax,        mm1

    mov         edx,        eax
    shr         edx,        16

    mov         WORD PTR[edi],   ax
    mov         BYTE PTR[edi+2], dl

  }

}

__declspec(align(16)) const static unsigned short one_thirds[] = {  85,  85,  85,  85 };
__declspec(align(16)) const static unsigned short two_thirds[] = { 171, 171, 171, 171 };

static
void vertical_band_5_3_scale_mmx(unsigned char *source, unsigned int src_pitch, unsigned char *dest, unsigned int dest_pitch, unsigned int dest_width) {

  __asm {
    push        ebx

    mov         esi,    source                    // Get the source and destination pointer
    mov         ecx,    src_pitch               // Get the pitch size

    mov         edi,    dest                    // tow lines below
    pxor        mm7,    mm7                     // clear out mm7

    mov         edx,    dest_pitch               // Loop counter
    movq        mm5,    one_thirds

    movq        mm6,    two_thirds
    mov         ebx,    dest_width;

    vs_5_3_loop:

    movd        mm0,    DWORD ptr [esi]         // src[0];
    movd        mm1,    DWORD ptr [esi+ecx]     // src[1];

    movd        mm2,    DWORD ptr [esi+ecx*2]
    lea         eax,    [esi+ecx*2]             //

    punpcklbw   mm1,    mm7
    punpcklbw   mm2,    mm7

    pmullw      mm1,    mm5
    pmullw      mm2,    mm6

    movd        mm3,    DWORD ptr [eax+ecx]
    movd        mm4,    DWORD ptr [eax+ecx*2]

    punpcklbw   mm3,    mm7
    punpcklbw   mm4,    mm7

    pmullw      mm3,    mm6
    pmullw      mm4,    mm5


    movd        DWORD PTR [edi], mm0
    paddw       mm1,    mm2

    paddw       mm1,    round_values
    psrlw       mm1,    8

    packuswb    mm1,    mm7
    paddw       mm3,    mm4

    paddw       mm3,    round_values
    movd        DWORD PTR [edi+edx], mm1

    psrlw       mm3,    8
    packuswb    mm3,    mm7

    movd        DWORD PTR [edi+edx*2], mm3


    add         edi,    4
    add         esi,    4

    sub         ebx,    4
    jg          vs_5_3_loop

    pop         ebx
  }
}




/****************************************************************************
 *
 *  ROUTINE       : horizontal_line_2_1_scale
 *
 *  INPUTS        : const unsigned char *source :
 *                  unsigned int source_width    :
 *                  unsigned char *dest         :
 *                  unsigned int dest_width      :
 *
 *  OUTPUTS       : None.
 *
 *  RETURNS       : void
 *
 *  FUNCTION      : 1 to 2 up-scaling of a horizontal line of pixels.
 *
 *  SPECIAL NOTES : None.
 *
 ****************************************************************************/
static
void horizontal_line_2_1_scale_mmx
(
  const unsigned char *source,
  unsigned int source_width,
  unsigned char *dest,
  unsigned int dest_width
) {
  (void) dest_width;
  (void) source_width;
  __asm {
    mov         esi,    source
    mov         edi,    dest

    pxor        mm7,    mm7
    mov         ecx,    dest_width

    xor         edx,    edx
    hs_2_1_loop:

    movq        mm0,    [esi+edx*2]
    psllw       mm0,    8

    psrlw       mm0,    8
    packuswb    mm0,    mm7

    movd        DWORD Ptr [edi+edx], mm0;
    add         edx,    4

    cmp         edx,    ecx
    jl          hs_2_1_loop

  }
}



static
void vertical_band_2_1_scale_mmx(unsigned char *source, unsigned int src_pitch, unsigned char *dest, unsigned int dest_pitch, unsigned int dest_width) {
  (void) dest_pitch;
  (void) src_pitch;
  vpx_memcpy(dest, source, dest_width);
}


__declspec(align(16)) const static unsigned short three_sixteenths[] = {  48,  48,  48,  48 };
__declspec(align(16)) const static unsigned short ten_sixteenths[]   = { 160, 160, 160, 160 };

static
void vertical_band_2_1_scale_i_mmx(unsigned char *source, unsigned int src_pitch, unsigned char *dest, unsigned int dest_pitch, unsigned int dest_width) {

  (void) dest_pitch;
  __asm {
    mov         esi,        source
    mov         edi,        dest

    mov         eax,        src_pitch
    mov         edx,        dest_width

    pxor        mm7,        mm7
    sub         esi,        eax             // back one line


    lea         ecx,        [esi+edx];
    movq        mm6,        round_values;

    movq        mm5,        three_sixteenths;
    movq        mm4,        ten_sixteenths;

    vs_2_1_i_loop:
    movd        mm0,        [esi]           //
    movd        mm1,        [esi+eax]       //

    movd        mm2,        [esi+eax*2]     //
    punpcklbw   mm0,        mm7

    pmullw      mm0,        mm5
    punpcklbw   mm1,        mm7

    pmullw      mm1,        mm4
    punpcklbw   mm2,        mm7

    pmullw      mm2,        mm5
    paddw       mm0,        round_values

    paddw       mm1,        mm2
    paddw       mm0,        mm1

    psrlw       mm0,        8
    packuswb    mm0,        mm7

    movd        DWORD PTR [edi],        mm0
    add         esi,        4

    add         edi,        4;
    cmp         esi,        ecx
    jl          vs_2_1_i_loop

  }
}



void
register_mmxscalers(void) {
  vp8_horizontal_line_1_2_scale        = horizontal_line_1_2_scale_mmx;
  vp8_vertical_band_1_2_scale          = vertical_band_1_2_scale_mmx;
  vp8_last_vertical_band_1_2_scale      = last_vertical_band_1_2_scale_mmx;
  vp8_horizontal_line_3_5_scale        = horizontal_line_3_5_scale_mmx;
  vp8_vertical_band_3_5_scale          = vertical_band_3_5_scale_mmx;
  vp8_last_vertical_band_3_5_scale      = last_vertical_band_3_5_scale_mmx;
  vp8_horizontal_line_4_5_scale        = horizontal_line_4_5_scale_mmx;
  vp8_vertical_band_4_5_scale          = vertical_band_4_5_scale_mmx;
  vp8_last_vertical_band_4_5_scale      = last_vertical_band_4_5_scale_mmx;

  vp8_horizontal_line_3_4_scale        = vp8cx_horizontal_line_3_4_scale_c;
  vp8_vertical_band_3_4_scale          = vp8cx_vertical_band_3_4_scale_c;
  vp8_last_vertical_band_3_4_scale      = vp8cx_last_vertical_band_3_4_scale_c;
  vp8_horizontal_line_2_3_scale        = vp8cx_horizontal_line_2_3_scale_c;
  vp8_vertical_band_2_3_scale          = vp8cx_vertical_band_2_3_scale_c;
  vp8_last_vertical_band_2_3_scale      = vp8cx_last_vertical_band_2_3_scale_c;



  vp8_vertical_band_5_4_scale           = vertical_band_5_4_scale_mmx;
  vp8_vertical_band_5_3_scale           = vertical_band_5_3_scale_mmx;
  vp8_vertical_band_2_1_scale           = vertical_band_2_1_scale_mmx;
  vp8_vertical_band_2_1_scale_i         = vertical_band_2_1_scale_i_mmx;
  vp8_horizontal_line_2_1_scale         = horizontal_line_2_1_scale_mmx;
  vp8_horizontal_line_5_3_scale         = horizontal_line_5_3_scale_mmx;
  vp8_horizontal_line_5_4_scale         = horizontal_line_5_4_scale_mmx;




}