-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathpragmarc-encryption-threefish-block_256.adb
185 lines (148 loc) · 7.49 KB
/
pragmarc-encryption-threefish-block_256.adb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
-- PragmAda Reusable Component (PragmARC)
-- Copyright (C) 2022 by PragmAda Software Engineering. All rights reserved.
-- Released under the terms of the BSD 3-Clause license; see https://opensource.org/licenses
-- **************************************************************************
--
-- 2022 Feb 01 J. Carter V1.2--Reorganization for 512- and 1024-bit versions
-- 2021 May 01 J. Carter V1.1--Adhere to coding standard
-- 2021 Feb 01 J. Carter V1.0--Initial PragmARC version
--
package body PragmARC.Encryption.Threefish.Block_256 is
use type Word;
procedure Create_Key_Schedule (Key : in Block; Tweak : in Couple; Key_Schedule : out Key_Schedule_Handle) is
C240 : constant := 16#1BD1_1BDA_A9FC_1A22#;
Modulus : constant := Num_Words + 1;
Ks : Key_Schedule_Handle renames Key_Schedule;
begin -- Create_Key_Schedule
Ks.Key (Key'Range) := Key;
Ks.Tweak (Tweak'Range) := Tweak;
Ks.Key (Ks.Key'Last) := C240;
Xor_All : for I in Key'Range loop
Ks.Key (Ks.Key'Last) := Ks.Key (Ks.Key'Last) xor Key (I);
end loop Xor_All;
Ks.Tweak (Ks.Tweak'Last) := Tweak (Tweak'First) xor Tweak (Tweak'Last);
All_Subkeys : for S in Ks.Subkey'Range loop
Ks.Subkey (S) (0) := Ks.Key (S rem Modulus);
Ks.Subkey (S) (1) := Ks.Key ( (S + 1) rem Modulus) + Ks.Tweak (S rem 3);
Ks.Subkey (S) (2) := Ks.Key ( (S + 2) rem Modulus) + Ks.Tweak ( (S + 1) rem 3);
Ks.Subkey (S) (3) := Ks.Key ( (S + 3) rem Modulus) + Word (S);
end loop All_Subkeys;
Ks.Valid := True;
end Create_Key_Schedule;
procedure Permute (Text : in out Block); -- Same for en- and de-cryption
type Pair_Id is (First, Last); -- 2 pairs per Block
type Rotation_Amount is array (0 .. 7, Pair_Id) of Positive; -- Indexed by Round mod 8
Rot : constant Rotation_Amount := (0 => (First => 14, Last => 16),
1 => (First => 52, Last => 57),
2 => (First => 23, Last => 40),
3 => (First => 5, Last => 37),
4 => (First => 25, Last => 33),
5 => (First => 46, Last => 12),
6 => (First => 58, Last => 22),
7 => (First => 32, Last => 32) );
procedure Encrypt (Key_Schedule : in Key_Schedule_Handle; Text : in out Block; Last_Round : in Round_ID := Round_ID'Last) is
function "+" (Left : Block; Right : Block) return Block; -- Word-by-word addition without carry
procedure Mix (Round : in Round_Id; Side : in Pair_Id; Pair : in out Couple); -- Performs the MIX operation in place
function "+" (Left : Block; Right : Block) return Block is
(Left (0) + Right (0), Left (1) + Right (1), Left (2) + Right (2), Left (3) + Right (3) );
procedure Mix (Round : in Round_Id; Side : in Pair_Id; Pair : in out Couple) is
-- Empty
begin -- Mix
Pair (Pair'First) := Pair (Pair'First) + Pair (Pair'Last);
Pair (Pair'Last) := Interfaces.Rotate_Left (Pair (Pair'Last), Rot (Round rem 8, Side) ) xor Pair (Pair'First);
end Mix;
Ks : Key_Schedule_Handle renames Key_Schedule;
begin -- Encrypt
All_Rounds : for Round in 0 .. Last_Round loop
if Round rem 4 = 0 then
Text := Text + Ks.Subkey (Round / 4);
end if;
Mix (Round => Round, Side => First, Pair => Text (0 .. 1) );
Mix (Round => Round, Side => Last, Pair => Text (2 .. 3) );
Permute (Text => Text);
end loop All_Rounds;
Text := Text + Ks.Subkey ( (Last_Round + 1) / 4);
end Encrypt;
procedure Decrypt (Key_Schedule : in Key_Schedule_Handle; Text : in out Block) is
function "-" (Left : Block; Right : Block) return Block; -- Word-by-word subtraction without borrow
procedure Unmix (Round : in Round_Id; Side : in Pair_Id; Pair : in out Couple); -- Inverse of the Mix procedure
function "-" (Left : Block; Right : Block) return Block is
(Left (0) - Right (0), Left (1) - Right (1), Left (2) - Right (2), Left (3) - Right (3) );
procedure Unmix (Round : in Round_Id; Side : in Pair_Id; Pair : in out Couple) is
-- Empty
begin -- Unmix
Pair (Pair'Last) := Interfaces.Rotate_Right (Pair (Pair'Last) xor Pair (Pair'First), Rot (Round rem 8, Side) );
Pair (Pair'First) := Pair (Pair'First) - Pair (Pair'Last);
end Unmix;
Ks : Key_Schedule_Handle renames Key_Schedule;
begin -- Decrypt
Text := Text - Ks.Subkey (Num_Rounds / 4);
All_Rounds : for Round in reverse Round_Id loop
Permute (Text => Text);
Unmix (Round => Round, Side => Last, Pair => Text (2 .. 3) );
UnmIx (Round => Round, Side => First, Pair => Text (0 .. 1) );
if Round rem 4 = 0 then
Text := Text - Ks.Subkey (Round / 4);
end if;
end loop All_Rounds;
end Decrypt;
Bytes_Per_Block : constant := 32;
function Encrypt (Key_Schedule : in Key_Schedule_Handle; Text : in Byte_List) return Block_List is
Num_Blocks : constant Natural := (Text'Length + Bytes_Per_Block - 1) / Bytes_Per_Block;
Slice : Block;
Result : Block_List (1 .. Num_Blocks);
Start : Positive := Text'First;
begin -- Encrypt
All_Blocks : for I in Result'Range loop
if Start + Bytes_Per_Block - 1 > Text'Last then
Slice := Block_From_Bytes (Text (Start .. Text'Last) & (Text'Last + 1 .. Start + Bytes_Per_Block - 1 => 0) );
else
Slice := Block_From_Bytes (Text (Start .. Start + Bytes_Per_Block - 1) );
end if;
Start := Start + Bytes_Per_Block;
Encrypt (Key_Schedule => Key_Schedule, Text => Slice);
Result (I) := Slice;
end loop All_Blocks;
return Result;
end Encrypt;
function Decrypt (Key_Schedule : in Key_Schedule_Handle; Text : in Block_List) return Byte_List is
Num_Bytes : constant Natural := Text'Length * Bytes_Per_Block;
Slice : Block;
Result : Byte_List (1 .. Num_Bytes);
Start : Positive := 1;
begin -- Decrypt
All_Blocks : for I in Text'Range loop
Slice := Text (I);
Decrypt (Key_Schedule => Key_Schedule, Text => Slice);
Result (Start .. Start + Bytes_Per_Block - 1) := Bytes_From_Block (Slice);
Start := Start + Bytes_Per_Block;
end loop All_Blocks;
return Result;
end Decrypt;
function Block_From_Bytes (List : in Block_As_Bytes) return Block is
Result : Block;
Start : Positive := List'First;
begin -- Block_From_Bytes
Convert : for I in Result'Range loop
Result (I) := Word_From_Bytes (List (Start .. Start + Word_As_Bytes'Length - 1) );
Start := Start + Word_As_Bytes'Length;
end loop Convert;
return Result;
end Block_From_Bytes;
function Bytes_From_Block (Value : in Block) return Block_As_Bytes is
Result : Block_As_Bytes;
Start : Positive := Result'First;
begin -- Bytes_From_Block
Convert : for I in Value'Range loop
Result (Start .. Start + Word_As_Bytes'Length - 1) := Bytes_From_Word (Value (I) );
Start := Start + Word_As_Bytes'Length;
end loop Convert;
return Result;
end Bytes_From_Block;
procedure Permute (Text : in out Block) is
Temp : constant Word := Text (1);
begin -- Permute
Text (1) := Text (3);
Text (3) := Temp;
end Permute;
end PragmARC.Encryption.Threefish.Block_256;