7
7
import org .bouncycastle .crypto .params .AEADParameters ;
8
8
import org .bouncycastle .crypto .params .KeyParameter ;
9
9
import org .bouncycastle .crypto .params .ParametersWithIV ;
10
+ import org .bouncycastle .util .Arrays ;
11
+ import org .bouncycastle .util .dispose .Disposable ;
12
+
13
+ import javax .security .auth .DestroyFailedException ;
14
+ import javax .security .auth .Destroyable ;
10
15
11
16
public class AESNativeGCMPacketCipher
12
- extends AESPacketCipherEngine
13
- implements AESGCMModePacketCipher
17
+ extends AESPacketCipherEngine
18
+ implements AESGCMModePacketCipher , Destroyable
14
19
{
15
- public static AESGCMModePacketCipher newInstance ()
16
- {
17
- return new AESNativeGCMPacketCipher ();
18
- }
19
20
20
- private AESNativeGCMPacketCipher ()
21
+
22
+ private byte [] lastKey ;
23
+ private byte [] lastNonce ;
24
+ private boolean destroyed ;
25
+
26
+ public AESNativeGCMPacketCipher ()
21
27
{
22
28
}
23
29
@@ -27,7 +33,7 @@ public int getOutputSize(boolean encryption, CipherParameters parameters, int le
27
33
int macSize ;
28
34
if (parameters instanceof AEADParameters )
29
35
{
30
- AEADParameters param = (AEADParameters )parameters ;
36
+ AEADParameters param = (AEADParameters ) parameters ;
31
37
int macSizeBits = param .getMacSize ();
32
38
if (macSizeBits < 32 || macSizeBits > 128 || (macSizeBits & 7 ) != 0 )
33
39
{
@@ -48,16 +54,17 @@ else if (parameters instanceof ParametersWithIV)
48
54
}
49
55
50
56
@ Override
51
- public int processPacket (boolean encryption , CipherParameters params , byte [] input , int inOff , int len , byte [] output , int outOff )
52
- throws PacketCipherException
57
+ public int processPacket (boolean encryption , CipherParameters params , byte [] input , int inOff , int len ,
58
+ byte [] output , int outOff )
59
+ throws PacketCipherException
53
60
{
54
61
int macSize ;
55
62
byte [] nonce ;
56
63
byte [] initialAssociatedText ;
57
64
byte [] key ;
58
65
if (params instanceof AEADParameters )
59
66
{
60
- AEADParameters param = (AEADParameters )params ;
67
+ AEADParameters param = (AEADParameters ) params ;
61
68
nonce = param .getNonce ();
62
69
initialAssociatedText = param .getAssociatedText ();
63
70
@@ -68,15 +75,40 @@ public int processPacket(boolean encryption, CipherParameters params, byte[] inp
68
75
}
69
76
70
77
macSize = macSizeBits >> 3 ;
78
+
71
79
key = param .getKey ().getKey ();
80
+
81
+ // This only works if you use the same instance of packet cipher
82
+ // It matches the existing behavior of the normal GCM implementation
83
+ if (encryption && Arrays .areEqual (key , lastKey ) && Arrays .areEqual (nonce , lastNonce ))
84
+ {
85
+ throw new IllegalArgumentException ("cannot reuse nonce for GCM encryption" );
86
+ }
87
+
88
+ lastKey = Arrays .clone (key );
89
+ lastNonce = Arrays .clone (nonce );
90
+
91
+
72
92
}
73
93
else if (params instanceof ParametersWithIV )
74
94
{
75
- ParametersWithIV param = (ParametersWithIV )params ;
95
+ ParametersWithIV param = (ParametersWithIV ) params ;
76
96
nonce = param .getIV ().clone ();
77
97
initialAssociatedText = null ;
78
98
macSize = 16 ;
79
- key = ((KeyParameter )param .getParameters ()).getKey ();
99
+
100
+ key = ((KeyParameter ) param .getParameters ()).getKey ();
101
+
102
+ // This only works if you use the same instance of packet cipher
103
+ // It matches the existing behavior of the normal GCM implementation
104
+ if (encryption && Arrays .areEqual (key , lastKey ) && Arrays .areEqual (nonce , lastNonce ))
105
+ {
106
+ throw new IllegalArgumentException ("cannot reuse nonce for GCM encryption" );
107
+ }
108
+
109
+ lastKey = Arrays .clone (key );
110
+ lastNonce = Arrays .clone (nonce );
111
+
80
112
}
81
113
else
82
114
{
@@ -88,7 +120,7 @@ else if (params instanceof ParametersWithIV)
88
120
try
89
121
{
90
122
result = processPacket (encryption , key , key .length , nonce , nonce .length , initialAssociatedText , iatLen ,
91
- macSize , input , inOff , len , output , outOff , outLen );
123
+ macSize , input , inOff , len , output , outOff , outLen );
92
124
}
93
125
catch (Exception e )
94
126
{
@@ -100,10 +132,27 @@ else if (params instanceof ParametersWithIV)
100
132
static native int getOutputSize (boolean encryption , int len , int macSize );
101
133
102
134
static native int processPacket (boolean encryption , byte [] key , int keyLen , byte [] nonce , int nonLen , byte [] aad ,
103
- int aadLen , int macSize , byte [] in , int inOff , int inLen , byte [] out , int outOff , int outLen );
135
+ int aadLen , int macSize , byte [] in , int inOff , int inLen , byte [] out , int outOff ,
136
+ int outLen );
137
+
104
138
@ Override
105
139
public String toString ()
106
140
{
107
141
return "GCM Packet Cipher (Native)" ;
108
142
}
143
+
144
+ @ Override
145
+ public void destroy () throws DestroyFailedException
146
+ {
147
+ Arrays .clear (lastKey );
148
+ Arrays .clear (lastNonce );
149
+ destroyed = true ;
150
+ }
151
+
152
+ @ Override
153
+ public boolean isDestroyed ()
154
+ {
155
+ return destroyed ;
156
+ }
157
+
109
158
}
0 commit comments