【发布时间】:2019-07-31 20:45:52
【问题描述】:
我使用的是 Delphi 10.3.2 我正在尝试将 DCPCrypt 单元与 firemonkey 框架一起使用来加密字符串。 它在 Win32、Win64 和 macOS 32 目标上 100% 运行,结果始终相同。但是当我为 macOS 64 编译时,结果就不同了。
这是使用的代码:
function EncodeAES(code:ansistring; key:ansistring):string;
var
s,u:ansistring;
enc: TEncoding;
k,iv, Data, Crypt: TBytes;
Cipher: TDCP_rijndael;
begin
u:='';
enc:=TEncoding.ANSI;
Data := enc.GetBytes(code); // VMpuXJGbUNOv
k:=enc.GetBytes(key); // kj3214ed)k32nre2
iv:=enc.GetBytes(u);
Cipher:=TDCP_rijndael.Create(nil);
Cipher.Init(K[0], 128, @iv[0]);
Crypt:=Copy(Data, 0, Length(Data));
BytePadding(Crypt, Cipher.BlockSize, pmPKCS7);
Cipher.EncryptECB(Crypt[0], Crypt[0]);
Cipher.Free;
s:=tencoding.ANSI.GetString(crypt);
result:=StringToHex(s);
end;
这是 BytePadding 函数:
procedure BytePadding(var Data: TBytes; BlockSize: integer; PaddingMode: TPaddingMode);
var
I, DataBlocks, DataLength, PaddingStart, PaddingCount: integer;
begin
BlockSize := BlockSize div 8;
if PaddingMode in [pmZeroPadding, pmRandomPadding] then
if Length(Data) mod BlockSize = 0 then Exit;
DataBlocks := (Length(Data) div BlockSize) + 1;
DataLength := DataBlocks * BlockSize;
PaddingCount := DataLength - Length(Data);
if PaddingMode in [pmANSIX923, pmISO10126, pmPKCS7] then
if PaddingCount > $FF then Exit;
PaddingStart := Length(Data);
SetLength(Data, DataLength);
case PaddingMode of
pmZeroPadding, pmANSIX923, pmISO7816: // fill with $00 bytes
FillChar(Data[PaddingStart], PaddingCount, 0);
pmPKCS7: // fill with PaddingCount bytes
FillChar(Data[PaddingStart], PaddingCount, PaddingCount);
pmRandomPadding, pmISO10126: // fill with random bytes
for I := PaddingStart to DataLength-1 do Data[I] := Random($FF);
end;
case PaddingMode of
pmANSIX923, pmISO10126:
Data[DataLength-1] := PaddingCount; // set end-marker with number of bytes added
pmISO7816:
Data[PaddingStart] := $80; // set fixed end-markder $80
end;
end;
我这样调用函数:
procedure TForm1.BtnClick(Sender: TObject);
var
s:string;
begin
s:=EncodeAES('VMpuXJGbUNOv','kj3214ed)k32nre2');
end;
Win32/Win64/macOS 32 的结果(正确): E32A9DE47CC60BDB70CA27885128D17A
macOS 64 的结果(错误): CF622155545E485AC3A083E8A0478493
我做错了什么?
【问题讨论】:
-
啊!!!不再。如此多的问题将二进制和文本视为可互换的。请让它停下来!不过说真的,加密算法是在二进制上运行的。您的加密函数需要使用字节数组作为输入和输出。
-
我假设当 data、crypt、iv 和 k 变量声明为 TBytes 时,它们是字节数组。
-
@Softtouch 是的,但您有责任在机器/平台上正确且一致地使用有效字节填充它们,而您的
EncodeAES()代码根本没有这样做。 -
DCPCrypt2 在 DCPcrypt2.pas 中的定义不正确。 dword = longword 应该是:dword = Cardinal。长字在 64 位 Posix 系统上是 8 个字节
-
@DaveNottage 我会选择
UInt32而不是Cardinal
标签: delphi firemonkey