你好,我已经开发了镜像/水平翻转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版本的整个代码,这方面的任何提示也会有所帮助..
答案 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
个字节复制到新文件中。向上移动一条扫描线并重复该过程,直到它们全部被复制。
编辑:在重新阅读您的问题后,我仍然不确定您正在翻转图像的方式,所以我不确定这是否适用。