Код Delphi / ASM несовместим с 64bit?
У меня есть пример исходного кода для OpenGL, я хотел скомпилировать 64-битную версию (используя Delphi XE2), но есть некоторый код ASM, который не компилируется, и я ничего не знаю об ASM. Вот код ниже, и я помещаю два сообщения об ошибках в строки, которые терпят неудачу...
// Copy a pixel from source to dest and Swap the RGB color values
procedure CopySwapPixel(const Source, Destination: Pointer);
asm
push ebx //[DCC Error]: E2116 Invalid combination of opcode and operands
mov bl,[eax+0]
mov bh,[eax+1]
mov [edx+2],bl
mov [edx+1],bh
mov bl,[eax+2]
mov bh,[eax+3]
mov [edx+0],bl
mov [edx+3],bh
pop ebx //[DCC Error]: E2116 Invalid combination of opcode and operands
end;
2 ответов
эта процедура меняет порядок байтов ABGR на ARGB и наоборот.
В 32bit этот код должен выполнять всю работу:
mov ecx, [eax] //ABGR from src
bswap ecx //RGBA
ror ecx, 8 //ARGB
mov [edx], ecx //to dest
правильный код для X64 -
mov ecx, [rcx] //ABGR from src
bswap ecx //RGBA
ror ecx, 8 //ARGB
mov [rdx], ecx //to dest
еще один вариант-сделать чистую версию Pascal, которая изменяет порядок байтов в представлении массива: 0123 на 2103 (swap 0th и 2th байты).
procedure Swp(const Source, Destination: Pointer);
var
s, d: PByteArray;
begin
s := PByteArray(Source);
d := PByteArray(Destination);
d[0] := s[2];
d[1] := s[1];
d[2] := s[0];
d[3] := s[3];
end;
64 бит имеет разные имена для регистров указателей, и это передается разница. Первые четыре параметра встроенных ассемблерных функций передаются через RCX, RDX, R8 и R9 соответственно
EBX -> RBX
EAX -> RAX
EDX -> RDX
попробуй такое
procedure CopySwapPixel(const Source, Destination: Pointer);
{$IFDEF CPUX64}
asm
mov al,[rcx+0]
mov ah,[rcx+1]
mov [rdx+2],al
mov [rdx+1],ah
mov al,[rcx+2]
mov ah,[rcx+3]
mov [rdx+0],al
mov [rdx+3],ah
end;
{$ELSE}
asm
push ebx //[DCC Error]: E2116 Invalid combination of opcode and operands
mov bl,[eax+0]
mov bh,[eax+1]
mov [edx+2],bl
mov [edx+1],bh
mov bl,[eax+2]
mov bh,[eax+3]
mov [edx+0],bl
mov [edx+3],bh
pop ebx //[DCC Error]: E2116 Invalid combination of opcode and operands
end;
{$ENDIF}