更改代码水平翻转“8bpp .bmp image”以在x86中水平翻转“1bpp .bmp image”

时间:2011-01-22 12:19:34

标签: assembly image-processing bitmap x86 x86-64

你好,我已经开发了镜像/水平翻转8 bpp .BMP图像的代码。正确处理任何宽度,不仅是4的倍数。现在我必须将此代码转换为相同但是为1 bpp。使用x86的bmp图像(灰度)。困难的部分是我不知道如何超出单个位可能有人可以编辑此代码..

        section     .text
    global      mirrorbmp8
mirrorbmp8:
    push        ebp
    mov     ebp, esp
    push        ebx
    push        esi
    push        edi

    mov     ebx, [ebp+12]       ;width - without padding
    and     ebx, 11b
    je      init            ;checking if there is a padding
    mov     edi, 4
    sub     edi, ebx
    add     [ebp+12], edi       ;width - with padding

init:
    mov     ebx, [ebp+16]   
    ;calculating the distance between top&bottom pixel
    dec     ebx
    mov     eax, [ebp+12]
    mul     ebx
    mov     esi, eax

    mov     edi, [ebp+8]    ;the first bottom pixel
    mov     edx, edi            ;the first top pixel
    mov     eax, edi
    add     eax, esi
    mov     ecx, [ebp+12]   
            ;register responsible for calc left columns

loop0:
    push        esi
    mov     esi, [ebp+12]

loop1:
    mov     bl, [edi]               ;changing pixels
    xchg        bl, [eax]
    mov     [edi], bl

    add     edi, esi        ;next pixel in this column
    sub     eax, esi
    cmp     edi, eax
    jl      loop1


    inc     edx             ;next bottom pixel
    mov     edi, edx

    mov     eax, edi                ;next top pixel
    pop     esi
    add     eax, esi

    dec     ecx         ;decrement number of columns left
    jnz     loop0           ;was that the last column?



end:
    pop     edi
    pop     esi 
    pop     ebx

    mov     esp, ebp
    pop     ebp
    ret

任何帮助将不胜感激。 在此先感谢:)

p.s如果我能够做这个版本,那么我也必须转换x86-64版本的整个代码,这方面的任何提示也会有所帮助..

2 个答案:

答案 0 :(得分:1)

如果你想知道如何从左到右而不是从上到下翻转图像,这就是你如何做到的。

首先,将位图的标题复制到另一个文件中。接下来,计算出每个扫描行末尾经过imageWidth%32的位数:

orphanBits = imageWidth % 32

在下面的示例中,orphanedBits为19.将扫描线末尾的最后两个DWORDS读入两个通用寄存器:

ebx = 10001010 11010101 00101010 10101010
eax = 01010101 01011000 00000000 00000000
  END OF SCAN LINE ^

使用SHRD操作数将位从ebx移到ecx,直到整个寄存器被填满:

shrd eax, ebx, orphanBits

ebx = 00000000 00000000 00010001 01011010
eax = 10100101 01010101 01001010 10101011
                       END OF SCAN LINE ^

然后使用以下代码交换eax

的位
mov edx,eax
shr eax,1
and edx,055555555h
and eax,055555555h
lea eax,[2*edx+eax]
mov edx,eax
shr eax,2
and edx,033333333h
and eax,033333333h
lea eax,[4*edx+eax]
mov edx,eax
shr eax,4
and edx,0F0F0F0Fh
and eax,0F0F0F0Fh
shl edx,4
add eax,edx
bswap eax

eax = 11010101 01010010 10101010 10100101
      ^ END OF SCAN LINE

将调整好的DWORD(现在按相反顺序)写入新图像。重复,直到读取整个扫描线。重复,直到读取所有扫描线。

编辑:在我记得交换字节而不是位之前,最初只有bswap。

答案 1 :(得分:0)

位图存储在扫描线中,每条扫描线有一行像素。你真正需要做的就是颠倒这些线的顺序。第一步是将位图的标题复制到另一个文件中。然后你应该计算每条扫描线的长度。因为每条扫描线总是32位填充,所以你需要一些数学运算:

scanlineLength = imageWidth / 8
IF imageWidth % 32 != 0
  scanlineLength += 1
ENDIF

接下来,将旧文件中的scanlineLength个字节复制到新文件中。向上移动一条扫描线并重复该过程,直到它们全部被复制。

编辑:在重新阅读您的问题后,我仍然不确定您正在翻转图像的方式,所以我不确定这是否适用。