2014-01-17 15:22:53 +00:00
|
|
|
{******************************************************************************}
|
|
|
|
{* DCPcrypt v2.0 written by David Barton (crypto@cityinthesky.co.uk) **********}
|
|
|
|
{******************************************************************************}
|
|
|
|
{* A binary compatible implementation of DES and Triple DES *******************}
|
|
|
|
{* Based on C source code by Eric Young ***************************************}
|
|
|
|
{******************************************************************************}
|
|
|
|
{* Copyright (c) 1999-2002 David Barton *}
|
|
|
|
{* Permission is hereby granted, free of charge, to any person obtaining a *}
|
|
|
|
{* copy of this software and associated documentation files (the "Software"), *}
|
|
|
|
{* to deal in the Software without restriction, including without limitation *}
|
|
|
|
{* the rights to use, copy, modify, merge, publish, distribute, sublicense, *}
|
|
|
|
{* and/or sell copies of the Software, and to permit persons to whom the *}
|
|
|
|
{* Software is furnished to do so, subject to the following conditions: *}
|
|
|
|
{* *}
|
|
|
|
{* The above copyright notice and this permission notice shall be included in *}
|
|
|
|
{* all copies or substantial portions of the Software. *}
|
|
|
|
{* *}
|
|
|
|
{* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR *}
|
|
|
|
{* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, *}
|
|
|
|
{* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL *}
|
|
|
|
{* THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER *}
|
|
|
|
{* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING *}
|
|
|
|
{* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER *}
|
|
|
|
{* DEALINGS IN THE SOFTWARE. *}
|
|
|
|
{******************************************************************************}
|
|
|
|
{******************************************************************************}
|
|
|
|
{* This implementation of DES is based on the C implementation by *}
|
|
|
|
{* Eric Young (eay@mincom.oz.au) *}
|
|
|
|
{******************************************************************************}
|
|
|
|
{* DES takes a 64bit key and discards every 8th bit (56bit effectively) *}
|
|
|
|
{* 3DES takes either a <= 128bit key and uses one key twice or takes a *}
|
|
|
|
{* <= 192bit key and uses each once (again discarding every 8th bit) *}
|
|
|
|
{******************************************************************************}
|
|
|
|
unit DCPdes;
|
|
|
|
|
|
|
|
{$MODE Delphi}
|
|
|
|
|
|
|
|
interface
|
|
|
|
uses
|
|
|
|
Classes, Sysutils, DCPcrypt2, DCPconst, DCPblockciphers;
|
|
|
|
|
|
|
|
type
|
|
|
|
TDCP_customdes= class(TDCP_blockcipher64)
|
|
|
|
protected
|
|
|
|
procedure DoInit(KeyB: PByteArray; KeyData: PDWordArray);
|
|
|
|
procedure EncryptBlock(const InData; var OutData; KeyData: PDWordArray);
|
|
|
|
procedure DecryptBlock(const InData; var OutData; KeyData: PDWordArray);
|
|
|
|
end;
|
|
|
|
|
|
|
|
type
|
|
|
|
TDCP_des= class(TDCP_customdes)
|
|
|
|
protected
|
|
|
|
KeyData: array[0..31] of dword;
|
|
|
|
procedure InitKey(const Key; Size: longword); override;
|
|
|
|
public
|
|
|
|
class function GetId: integer; override;
|
|
|
|
class function GetAlgorithm: string; override;
|
|
|
|
class function GetMaxKeySize: integer; override;
|
|
|
|
class function SelfTest: boolean; override;
|
|
|
|
procedure Burn; override;
|
|
|
|
procedure EncryptECB(const InData; var OutData); override;
|
|
|
|
procedure DecryptECB(const InData; var OutData); override;
|
|
|
|
end;
|
|
|
|
|
|
|
|
TDCP_3des= class(TDCP_customdes)
|
|
|
|
protected
|
|
|
|
KeyData: array[0..2,0..31] of dword;
|
|
|
|
procedure InitKey(const Key; Size: longword); override;
|
|
|
|
public
|
|
|
|
class function GetId: integer; override;
|
|
|
|
class function GetAlgorithm: string; override;
|
|
|
|
class function GetMaxKeySize: integer; override;
|
|
|
|
class function SelfTest: boolean; override;
|
|
|
|
procedure Burn; override;
|
|
|
|
procedure EncryptECB(const InData; var OutData); override;
|
|
|
|
procedure DecryptECB(const InData; var OutData); override;
|
|
|
|
end;
|
|
|
|
|
|
|
|
{******************************************************************************}
|
|
|
|
{******************************************************************************}
|
|
|
|
implementation
|
|
|
|
{$R-}{$Q-}
|
|
|
|
|
|
|
|
{$I DCPdes.inc}
|
|
|
|
|
|
|
|
procedure hperm_op(var a, t: dword; n, m: dword);
|
|
|
|
begin
|
|
|
|
t:= ((a shl (16 - n)) xor a) and m;
|
|
|
|
a:= a xor t xor (t shr (16 - n));
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure perm_op(var a, b, t: dword; n, m: dword);
|
|
|
|
begin
|
|
|
|
t:= ((a shr n) xor b) and m;
|
|
|
|
b:= b xor t;
|
|
|
|
a:= a xor (t shl n);
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TDCP_customdes.DoInit(KeyB: PByteArray; KeyData: PDwordArray);
|
|
|
|
var
|
|
|
|
c, d, t, s, t2, i: dword;
|
|
|
|
begin
|
|
|
|
t := 0;
|
|
|
|
c:= KeyB^[0] or (KeyB^[1] shl 8) or (KeyB^[2] shl 16) or (KeyB^[3] shl 24);
|
|
|
|
d:= KeyB^[4] or (KeyB^[5] shl 8) or (KeyB^[6] shl 16) or (KeyB^[7] shl 24);
|
|
|
|
perm_op(d,c,t,4,$0f0f0f0f);
|
|
|
|
hperm_op(c,t,dword(-2),$cccc0000);
|
|
|
|
hperm_op(d,t,dword(-2),$cccc0000);
|
|
|
|
perm_op(d,c,t,1,$55555555);
|
|
|
|
perm_op(c,d,t,8,$00ff00ff);
|
|
|
|
perm_op(d,c,t,1,$55555555);
|
|
|
|
d:= ((d and $ff) shl 16) or (d and $ff00) or ((d and $ff0000) shr 16) or
|
|
|
|
((c and $f0000000) shr 4);
|
|
|
|
c:= c and $fffffff;
|
|
|
|
for i:= 0 to 15 do
|
|
|
|
begin
|
|
|
|
if shifts2[i]<> 0 then
|
|
|
|
begin
|
|
|
|
c:= ((c shr 2) or (c shl 26));
|
|
|
|
d:= ((d shr 2) or (d shl 26));
|
|
|
|
end
|
|
|
|
else
|
|
|
|
begin
|
|
|
|
c:= ((c shr 1) or (c shl 27));
|
|
|
|
d:= ((d shr 1) or (d shl 27));
|
|
|
|
end;
|
|
|
|
c:= c and $fffffff;
|
|
|
|
d:= d and $fffffff;
|
|
|
|
s:= des_skb[0,c and $3f] or
|
|
|
|
des_skb[1,((c shr 6) and $03) or ((c shr 7) and $3c)] or
|
|
|
|
des_skb[2,((c shr 13) and $0f) or ((c shr 14) and $30)] or
|
|
|
|
des_skb[3,((c shr 20) and $01) or ((c shr 21) and $06) or ((c shr 22) and $38)];
|
|
|
|
t:= des_skb[4,d and $3f] or
|
|
|
|
des_skb[5,((d shr 7) and $03) or ((d shr 8) and $3c)] or
|
|
|
|
des_skb[6, (d shr 15) and $3f ] or
|
|
|
|
des_skb[7,((d shr 21) and $0f) or ((d shr 22) and $30)];
|
|
|
|
t2:= ((t shl 16) or (s and $ffff));
|
|
|
|
KeyData^[(i shl 1)]:= ((t2 shl 2) or (t2 shr 30));
|
|
|
|
t2:= ((s shr 16) or (t and $ffff0000));
|
|
|
|
KeyData^[(i shl 1)+1]:= ((t2 shl 6) or (t2 shr 26));
|
|
|
|
end;
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TDCP_customdes.EncryptBlock(const InData; var OutData; KeyData: PDWordArray);
|
|
|
|
var
|
|
|
|
l, r, t, u: dword;
|
|
|
|
i: longint;
|
|
|
|
begin
|
|
|
|
r:= PDword(@InData)^;
|
|
|
|
l:= PDword(pointer(@InData)+4)^;
|
|
|
|
t:= ((l shr 4) xor r) and $0f0f0f0f;
|
|
|
|
r:= r xor t;
|
|
|
|
l:= l xor (t shl 4);
|
|
|
|
t:= ((r shr 16) xor l) and $0000ffff;
|
|
|
|
l:= l xor t;
|
|
|
|
r:= r xor (t shl 16);
|
|
|
|
t:= ((l shr 2) xor r) and $33333333;
|
|
|
|
r:= r xor t;
|
|
|
|
l:= l xor (t shl 2);
|
|
|
|
t:= ((r shr 8) xor l) and $00ff00ff;
|
|
|
|
l:= l xor t;
|
|
|
|
r:= r xor (t shl 8);
|
|
|
|
t:= ((l shr 1) xor r) and $55555555;
|
|
|
|
r:= r xor t;
|
|
|
|
l:= l xor (t shl 1);
|
|
|
|
r:= (r shr 29) or (r shl 3);
|
|
|
|
l:= (l shr 29) or (l shl 3);
|
|
|
|
i:= 0;
|
|
|
|
while i< 32 do
|
|
|
|
begin
|
|
|
|
u:= r xor KeyData^[i ];
|
|
|
|
t:= r xor KeyData^[i+1];
|
|
|
|
t:= (t shr 4) or (t shl 28);
|
|
|
|
l:= l xor des_SPtrans[0,(u shr 2) and $3f] xor
|
|
|
|
des_SPtrans[2,(u shr 10) and $3f] xor
|
|
|
|
des_SPtrans[4,(u shr 18) and $3f] xor
|
|
|
|
des_SPtrans[6,(u shr 26) and $3f] xor
|
|
|
|
des_SPtrans[1,(t shr 2) and $3f] xor
|
|
|
|
des_SPtrans[3,(t shr 10) and $3f] xor
|
|
|
|
des_SPtrans[5,(t shr 18) and $3f] xor
|
|
|
|
des_SPtrans[7,(t shr 26) and $3f];
|
|
|
|
u:= l xor KeyData^[i+2];
|
|
|
|
t:= l xor KeyData^[i+3];
|
|
|
|
t:= (t shr 4) or (t shl 28);
|
|
|
|
r:= r xor des_SPtrans[0,(u shr 2) and $3f] xor
|
|
|
|
des_SPtrans[2,(u shr 10) and $3f] xor
|
|
|
|
des_SPtrans[4,(u shr 18) and $3f] xor
|
|
|
|
des_SPtrans[6,(u shr 26) and $3f] xor
|
|
|
|
des_SPtrans[1,(t shr 2) and $3f] xor
|
|
|
|
des_SPtrans[3,(t shr 10) and $3f] xor
|
|
|
|
des_SPtrans[5,(t shr 18) and $3f] xor
|
|
|
|
des_SPtrans[7,(t shr 26) and $3f];
|
|
|
|
u:= r xor KeyData^[i+4];
|
|
|
|
t:= r xor KeyData^[i+5];
|
|
|
|
t:= (t shr 4) or (t shl 28);
|
|
|
|
l:= l xor des_SPtrans[0,(u shr 2) and $3f] xor
|
|
|
|
des_SPtrans[2,(u shr 10) and $3f] xor
|
|
|
|
des_SPtrans[4,(u shr 18) and $3f] xor
|
|
|
|
des_SPtrans[6,(u shr 26) and $3f] xor
|
|
|
|
des_SPtrans[1,(t shr 2) and $3f] xor
|
|
|
|
des_SPtrans[3,(t shr 10) and $3f] xor
|
|
|
|
des_SPtrans[5,(t shr 18) and $3f] xor
|
|
|
|
des_SPtrans[7,(t shr 26) and $3f];
|
|
|
|
u:= l xor KeyData^[i+6];
|
|
|
|
t:= l xor KeyData^[i+7];
|
|
|
|
t:= (t shr 4) or (t shl 28);
|
|
|
|
r:= r xor des_SPtrans[0,(u shr 2) and $3f] xor
|
|
|
|
des_SPtrans[2,(u shr 10) and $3f] xor
|
|
|
|
des_SPtrans[4,(u shr 18) and $3f] xor
|
|
|
|
des_SPtrans[6,(u shr 26) and $3f] xor
|
|
|
|
des_SPtrans[1,(t shr 2) and $3f] xor
|
|
|
|
des_SPtrans[3,(t shr 10) and $3f] xor
|
|
|
|
des_SPtrans[5,(t shr 18) and $3f] xor
|
|
|
|
des_SPtrans[7,(t shr 26) and $3f];
|
|
|
|
Inc(i,8);
|
|
|
|
end;
|
|
|
|
r:= (r shr 3) or (r shl 29);
|
|
|
|
l:= (l shr 3) or (l shl 29);
|
|
|
|
t:= ((r shr 1) xor l) and $55555555;
|
|
|
|
l:= l xor t;
|
|
|
|
r:= r xor (t shl 1);
|
|
|
|
t:= ((l shr 8) xor r) and $00ff00ff;
|
|
|
|
r:= r xor t;
|
|
|
|
l:= l xor (t shl 8);
|
|
|
|
t:= ((r shr 2) xor l) and $33333333;
|
|
|
|
l:= l xor t;
|
|
|
|
r:= r xor (t shl 2);
|
|
|
|
t:= ((l shr 16) xor r) and $0000ffff;
|
|
|
|
r:= r xor t;
|
|
|
|
l:= l xor (t shl 16);
|
|
|
|
t:= ((r shr 4) xor l) and $0f0f0f0f;
|
|
|
|
l:= l xor t;
|
|
|
|
r:= r xor (t shl 4);
|
|
|
|
PDword(@OutData)^:= l;
|
|
|
|
PDword(pointer(@OutData)+4)^:= r;
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TDCP_customdes.DecryptBlock(const InData; var OutData; KeyData: PDWordArray);
|
|
|
|
var
|
|
|
|
l, r, t, u: dword;
|
|
|
|
i: longint;
|
|
|
|
begin
|
|
|
|
r:= PDword(@InData)^;
|
|
|
|
l:= PDword(pointer(@InData)+4)^;
|
|
|
|
t:= ((l shr 4) xor r) and $0f0f0f0f;
|
|
|
|
r:= r xor t;
|
|
|
|
l:= l xor (t shl 4);
|
|
|
|
t:= ((r shr 16) xor l) and $0000ffff;
|
|
|
|
l:= l xor t;
|
|
|
|
r:= r xor (t shl 16);
|
|
|
|
t:= ((l shr 2) xor r) and $33333333;
|
|
|
|
r:= r xor t;
|
|
|
|
l:= l xor (t shl 2);
|
|
|
|
t:= ((r shr 8) xor l) and $00ff00ff;
|
|
|
|
l:= l xor t;
|
|
|
|
r:= r xor (t shl 8);
|
|
|
|
t:= ((l shr 1) xor r) and $55555555;
|
|
|
|
r:= r xor t;
|
|
|
|
l:= l xor (t shl 1);
|
|
|
|
r:= (r shr 29) or (r shl 3);
|
|
|
|
l:= (l shr 29) or (l shl 3);
|
|
|
|
i:= 30;
|
|
|
|
while i> 0 do
|
|
|
|
begin
|
|
|
|
u:= r xor KeyData^[i ];
|
|
|
|
t:= r xor KeyData^[i+1];
|
|
|
|
t:= (t shr 4) or (t shl 28);
|
|
|
|
l:= l xor des_SPtrans[0,(u shr 2) and $3f] xor
|
|
|
|
des_SPtrans[2,(u shr 10) and $3f] xor
|
|
|
|
des_SPtrans[4,(u shr 18) and $3f] xor
|
|
|
|
des_SPtrans[6,(u shr 26) and $3f] xor
|
|
|
|
des_SPtrans[1,(t shr 2) and $3f] xor
|
|
|
|
des_SPtrans[3,(t shr 10) and $3f] xor
|
|
|
|
des_SPtrans[5,(t shr 18) and $3f] xor
|
|
|
|
des_SPtrans[7,(t shr 26) and $3f];
|
|
|
|
u:= l xor KeyData^[i-2];
|
|
|
|
t:= l xor KeyData^[i-1];
|
|
|
|
t:= (t shr 4) or (t shl 28);
|
|
|
|
r:= r xor des_SPtrans[0,(u shr 2) and $3f] xor
|
|
|
|
des_SPtrans[2,(u shr 10) and $3f] xor
|
|
|
|
des_SPtrans[4,(u shr 18) and $3f] xor
|
|
|
|
des_SPtrans[6,(u shr 26) and $3f] xor
|
|
|
|
des_SPtrans[1,(t shr 2) and $3f] xor
|
|
|
|
des_SPtrans[3,(t shr 10) and $3f] xor
|
|
|
|
des_SPtrans[5,(t shr 18) and $3f] xor
|
|
|
|
des_SPtrans[7,(t shr 26) and $3f];
|
|
|
|
u:= r xor KeyData^[i-4];
|
|
|
|
t:= r xor KeyData^[i-3];
|
|
|
|
t:= (t shr 4) or (t shl 28);
|
|
|
|
l:= l xor des_SPtrans[0,(u shr 2) and $3f] xor
|
|
|
|
des_SPtrans[2,(u shr 10) and $3f] xor
|
|
|
|
des_SPtrans[4,(u shr 18) and $3f] xor
|
|
|
|
des_SPtrans[6,(u shr 26) and $3f] xor
|
|
|
|
des_SPtrans[1,(t shr 2) and $3f] xor
|
|
|
|
des_SPtrans[3,(t shr 10) and $3f] xor
|
|
|
|
des_SPtrans[5,(t shr 18) and $3f] xor
|
|
|
|
des_SPtrans[7,(t shr 26) and $3f];
|
|
|
|
u:= l xor KeyData^[i-6];
|
|
|
|
t:= l xor KeyData^[i-5];
|
|
|
|
t:= (t shr 4) or (t shl 28);
|
|
|
|
r:= r xor des_SPtrans[0,(u shr 2) and $3f] xor
|
|
|
|
des_SPtrans[2,(u shr 10) and $3f] xor
|
|
|
|
des_SPtrans[4,(u shr 18) and $3f] xor
|
|
|
|
des_SPtrans[6,(u shr 26) and $3f] xor
|
|
|
|
des_SPtrans[1,(t shr 2) and $3f] xor
|
|
|
|
des_SPtrans[3,(t shr 10) and $3f] xor
|
|
|
|
des_SPtrans[5,(t shr 18) and $3f] xor
|
|
|
|
des_SPtrans[7,(t shr 26) and $3f];
|
|
|
|
Dec(i,8);
|
|
|
|
end;
|
|
|
|
r:= (r shr 3) or (r shl 29);
|
|
|
|
l:= (l shr 3) or (l shl 29);
|
|
|
|
t:= ((r shr 1) xor l) and $55555555;
|
|
|
|
l:= l xor t;
|
|
|
|
r:= r xor (t shl 1);
|
|
|
|
t:= ((l shr 8) xor r) and $00ff00ff;
|
|
|
|
r:= r xor t;
|
|
|
|
l:= l xor (t shl 8);
|
|
|
|
t:= ((r shr 2) xor l) and $33333333;
|
|
|
|
l:= l xor t;
|
|
|
|
r:= r xor (t shl 2);
|
|
|
|
t:= ((l shr 16) xor r) and $0000ffff;
|
|
|
|
r:= r xor t;
|
|
|
|
l:= l xor (t shl 16);
|
|
|
|
t:= ((r shr 4) xor l) and $0f0f0f0f;
|
|
|
|
l:= l xor t;
|
|
|
|
r:= r xor (t shl 4);
|
|
|
|
PDword(@OutData)^:= l;
|
|
|
|
PDword(pointer(@OutData)+4)^:= r;
|
|
|
|
end;
|
|
|
|
|
|
|
|
class function TDCP_des.GetMaxKeySize: integer;
|
|
|
|
begin
|
|
|
|
Result:= 64;
|
|
|
|
end;
|
|
|
|
|
|
|
|
class function TDCP_des.GetID: integer;
|
|
|
|
begin
|
|
|
|
Result:= DCP_des;
|
|
|
|
end;
|
|
|
|
|
|
|
|
class function TDCP_des.GetAlgorithm: string;
|
|
|
|
begin
|
|
|
|
Result:= 'DES';
|
|
|
|
end;
|
|
|
|
|
|
|
|
class function TDCP_des.SelfTest: boolean;
|
|
|
|
const
|
|
|
|
InData1: array[0..7] of byte=
|
|
|
|
($07,$56,$D8,$E0,$77,$47,$61,$D2);
|
|
|
|
OutData1: array[0..7] of byte=
|
|
|
|
($0C,$D3,$DA,$02,$00,$21,$DC,$09);
|
|
|
|
Key1: array[0..7] of byte=
|
|
|
|
($01,$70,$F1,$75,$46,$8F,$B5,$E6);
|
|
|
|
InData2: array[0..7] of byte=
|
|
|
|
($48,$0D,$39,$00,$6E,$E7,$62,$F2);
|
|
|
|
OutData2: array[0..7] of byte=
|
|
|
|
($A1,$F9,$91,$55,$41,$02,$0B,$56);
|
|
|
|
Key2: array[0..7] of byte=
|
|
|
|
($02,$58,$16,$16,$46,$29,$B0,$07);
|
|
|
|
var
|
|
|
|
Cipher: TDCP_des;
|
|
|
|
Data: array[0..7] of byte;
|
|
|
|
begin
|
|
|
|
dcpFillChar(Data, SizeOf(Data), 0);
|
|
|
|
Cipher:= TDCP_des.Create(nil);
|
|
|
|
Cipher.Init(Key1,Sizeof(Key1)*8,nil);
|
|
|
|
Cipher.EncryptECB(InData1,Data);
|
|
|
|
Result:= boolean(CompareMem(@Data,@OutData1,Sizeof(Data)));
|
|
|
|
Cipher.DecryptECB(Data,Data);
|
|
|
|
Result:= Result and boolean(CompareMem(@Data,@InData1,Sizeof(Data)));
|
|
|
|
Cipher.Burn;
|
|
|
|
Cipher.Init(Key2,Sizeof(Key2)*8,nil);
|
|
|
|
Cipher.EncryptECB(InData2,Data);
|
|
|
|
Result:= Result and boolean(CompareMem(@Data,@OutData2,Sizeof(Data)));
|
|
|
|
Cipher.DecryptECB(Data,Data);
|
|
|
|
Result:= Result and boolean(CompareMem(@Data,@InData2,Sizeof(Data)));
|
|
|
|
Cipher.Burn;
|
|
|
|
Cipher.Free;
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TDCP_des.InitKey(const Key; Size: longword);
|
|
|
|
var
|
|
|
|
KeyB: array[0..7] of byte;
|
|
|
|
begin
|
|
|
|
dcpFillChar(KeyB,Sizeof(KeyB),0);
|
|
|
|
Move(Key,KeyB,Size div 8);
|
|
|
|
DoInit(@KeyB,@KeyData);
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TDCP_des.Burn;
|
|
|
|
begin
|
|
|
|
FillChar(KeyData,Sizeof(KeyData),0);
|
|
|
|
inherited Burn;
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TDCP_des.EncryptECB(const InData; var OutData);
|
|
|
|
begin
|
|
|
|
if not fInitialized then
|
|
|
|
raise EDCP_blockcipher.Create('Cipher not initialized');
|
|
|
|
EncryptBlock(InData,OutData,@KeyData);
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TDCP_des.DecryptECB(const InData; var OutData);
|
|
|
|
begin
|
|
|
|
if not fInitialized then
|
|
|
|
raise EDCP_blockcipher.Create('Cipher not initialized');
|
|
|
|
DecryptBlock(InData,OutData,@KeyData);
|
|
|
|
end;
|
|
|
|
|
|
|
|
{******************************************************************************}
|
|
|
|
class function TDCP_3des.GetMaxKeySize: integer;
|
|
|
|
begin
|
|
|
|
Result:= 192;
|
|
|
|
end;
|
|
|
|
|
|
|
|
class function TDCP_3des.GetID: integer;
|
|
|
|
begin
|
|
|
|
Result:= DCP_3des;
|
|
|
|
end;
|
|
|
|
|
|
|
|
class function TDCP_3des.GetAlgorithm: string;
|
|
|
|
begin
|
|
|
|
Result:= '3DES';
|
|
|
|
end;
|
|
|
|
|
|
|
|
class function TDCP_3des.SelfTest: boolean;
|
|
|
|
const
|
|
|
|
Key: array[0..23] of byte=
|
|
|
|
($01,$23,$45,$67,$89,$ab,$cd,$ef,$fe,$dc,$ba,$98,
|
|
|
|
$76,$54,$32,$10,$89,$ab,$cd,$ef,$01,$23,$45,$67);
|
|
|
|
PlainText: array[0..7] of byte=
|
|
|
|
($01,$23,$45,$67,$89,$ab,$cd,$e7);
|
|
|
|
CipherText: array[0..7] of byte=
|
|
|
|
($de,$0b,$7c,$06,$ae,$5e,$0e,$d5);
|
|
|
|
var
|
|
|
|
Cipher: TDCP_3des;
|
|
|
|
Block: array[0..7] of byte;
|
|
|
|
begin
|
|
|
|
dcpFillChar(Block, SizeOf(Block), 0);
|
|
|
|
Cipher:= TDCP_3des.Create(nil);
|
|
|
|
Cipher.Init(Key,Sizeof(Key)*8,nil);
|
|
|
|
Cipher.EncryptECB(PlainText,Block);
|
|
|
|
Result:= CompareMem(@Block,@CipherText,Sizeof(CipherText));
|
|
|
|
Cipher.DecryptECB(Block,Block);
|
|
|
|
Result:= Result and CompareMem(@Block,@PlainText,Sizeof(PlainText));
|
|
|
|
Cipher.Free;
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TDCP_3des.InitKey(const Key; Size: longword);
|
|
|
|
var
|
|
|
|
KeyB: array[0..2,0..7] of byte;
|
|
|
|
begin
|
|
|
|
dcpFillChar(KeyB,Sizeof(KeyB),0);
|
|
|
|
Move(Key,KeyB,Size div 8);
|
|
|
|
DoInit(@KeyB[0],@KeyData[0]);
|
|
|
|
DoInit(@KeyB[1],@KeyData[1]);
|
|
|
|
if Size> 128 then
|
|
|
|
DoInit(@KeyB[2],@KeyData[2])
|
|
|
|
else
|
|
|
|
Move(KeyData[0],KeyData[2],128);
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TDCP_3des.Burn;
|
|
|
|
begin
|
|
|
|
FillChar(KeyData,Sizeof(KeyData),0);
|
|
|
|
inherited Burn;
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TDCP_3des.EncryptECB(const InData; var OutData);
|
|
|
|
begin
|
|
|
|
if not fInitialized then
|
|
|
|
raise EDCP_blockcipher.Create('Cipher not initialized');
|
|
|
|
EncryptBlock(InData,OutData,@KeyData[0]);
|
|
|
|
DecryptBlock(OutData,OutData,@KeyData[1]);
|
|
|
|
EncryptBlock(OutData,OutData,@KeyData[2]);
|
|
|
|
end;
|
|
|
|
|
|
|
|
procedure TDCP_3des.DecryptECB(const InData; var OutData);
|
|
|
|
begin
|
|
|
|
if not fInitialized then
|
|
|
|
raise EDCP_blockcipher.Create('Cipher not initialized');
|
|
|
|
DecryptBlock(InData,OutData,@KeyData[2]);
|
|
|
|
EncryptBlock(OutData,OutData,@KeyData[1]);
|
|
|
|
DecryptBlock(OutData,OutData,@KeyData[0]);
|
|
|
|
end;
|
|
|
|
|
|
|
|
|
|
|
|
end.
|