|
1 | 1 | package org.bouncycastle.pqc.jcajce.provider.ntruprime;
|
2 | 2 |
|
3 |
| -import org.bouncycastle.asn1.bc.BCObjectIdentifiers; |
4 |
| -import org.bouncycastle.asn1.x509.AlgorithmIdentifier; |
5 | 3 | import org.bouncycastle.crypto.SecretWithEncapsulation;
|
6 | 4 | import org.bouncycastle.crypto.Wrapper;
|
7 |
| -import org.bouncycastle.crypto.params.KeyParameter; |
8 | 5 | import org.bouncycastle.jcajce.spec.KTSParameterSpec;
|
9 | 6 | import org.bouncycastle.pqc.crypto.ntruprime.SNTRUPrimeKEMGenerator;
|
10 | 7 | import org.bouncycastle.pqc.jcajce.provider.util.WrapUtil;
|
|
13 | 10 | import javax.crypto.KEM;
|
14 | 11 | import javax.crypto.KEMSpi;
|
15 | 12 | import javax.crypto.spec.SecretKeySpec;
|
16 |
| -import java.security.InvalidAlgorithmParameterException; |
17 | 13 | import java.security.InvalidKeyException;
|
18 |
| -import java.security.KeyPair; |
19 | 14 | import java.security.SecureRandom;
|
20 | 15 | import java.util.Objects;
|
21 | 16 |
|
22 |
| -import static org.bouncycastle.pqc.jcajce.provider.Util.makeKeyBytes; |
23 |
| - |
24 | 17 | class SNTRUPrimeEncapsulatorSpi
|
25 | 18 | implements KEMSpi.EncapsulatorSpi
|
26 | 19 | {
|
27 | 20 | private final BCSNTRUPrimePublicKey publicKey;
|
28 | 21 | private final KTSParameterSpec parameterSpec;
|
29 |
| - private final SecureRandom random; |
30 | 22 | private final SNTRUPrimeKEMGenerator kemGen;
|
31 | 23 |
|
32 | 24 |
|
33 | 25 | public SNTRUPrimeEncapsulatorSpi(BCSNTRUPrimePublicKey publicKey, KTSParameterSpec parameterSpec, SecureRandom random)
|
34 | 26 | {
|
35 | 27 | this.publicKey = publicKey;
|
36 | 28 | this.parameterSpec = parameterSpec;
|
37 |
| - this.random = random; |
38 | 29 |
|
39 | 30 | kemGen = new SNTRUPrimeKEMGenerator(random);
|
40 | 31 | }
|
41 | 32 |
|
42 | 33 | @Override
|
43 | 34 | public KEM.Encapsulated engineEncapsulate(int from, int to, String algorithm)
|
44 | 35 | {
|
45 |
| - Objects.checkFromToIndex(from, to, engineEncapsulationSize()); |
| 36 | + Objects.checkFromToIndex(from, to, engineSecretSize()); |
46 | 37 | Objects.requireNonNull(algorithm, "null algorithm");
|
47 | 38 |
|
48 |
| - KTSParameterSpec.Builder builder = new KTSParameterSpec.Builder(parameterSpec.getKeyAlgorithmName(), parameterSpec.getKeySize()); |
| 39 | + // if algorithm is Generic then use parameterSpec to wrap key |
| 40 | + if (!parameterSpec.getKeyAlgorithmName().equals("Generic") && |
| 41 | + algorithm.equals("Generic")) |
| 42 | + { |
| 43 | + algorithm = parameterSpec.getKeyAlgorithmName(); |
| 44 | + } |
49 | 45 |
|
50 |
| - if (!algorithm.equals("Generic")) |
| 46 | + // check spec algorithm mismatch provided algorithm |
| 47 | + if (!parameterSpec.getKeyAlgorithmName().equals("Generic") && |
| 48 | + !parameterSpec.getKeyAlgorithmName().equals(algorithm)) |
51 | 49 | {
|
52 |
| - //TODO: |
53 |
| -// builder.withKdfAlgorithm(AlgorithmIdentifier.getInstance(algorithm)); |
| 50 | + throw new UnsupportedOperationException(parameterSpec.getKeyAlgorithmName() + " does not match " + algorithm); |
54 | 51 | }
|
55 |
| - KTSParameterSpec spec = builder.build(); |
| 52 | + |
| 53 | + // Only use KDF when ktsParameterSpec is provided |
| 54 | + // Considering any ktsParameterSpec with "Generic" as ktsParameterSpec not provided |
| 55 | + boolean wrapKey = !(parameterSpec.getKeyAlgorithmName().equals("Generic") && algorithm.equals("Generic")); |
56 | 56 |
|
57 | 57 | SecretWithEncapsulation secEnc = kemGen.generateEncapsulated(publicKey.getKeyParams());
|
58 | 58 |
|
59 | 59 | byte[] encapsulation = secEnc.getEncapsulation();
|
60 | 60 | byte[] secret = secEnc.getSecret();
|
61 | 61 |
|
62 |
| - byte[] kdfKey = Arrays.copyOfRange(secret, from, to); |
| 62 | + byte[] secretKey = Arrays.copyOfRange(secret, from, to); |
63 | 63 |
|
64 |
| - try |
| 64 | + if (wrapKey) |
65 | 65 | {
|
66 |
| - return new KEM.Encapsulated(new SecretKeySpec(makeKeyBytes(spec, kdfKey), algorithm), encapsulation , null); |
67 |
| - } |
68 |
| - catch (InvalidKeyException e) |
69 |
| - { |
70 |
| - throw new RuntimeException(e); |
| 66 | + try |
| 67 | + { |
| 68 | + KTSParameterSpec spec = parameterSpec; |
| 69 | + // Generate a new ktsParameterSpec if spec is generic but algorithm is not generic |
| 70 | + if (parameterSpec.getKeyAlgorithmName().equals("Generic")) |
| 71 | + { |
| 72 | + spec = new KTSParameterSpec.Builder(algorithm, secretKey.length * 8).withNoKdf().build(); |
| 73 | + } |
| 74 | + |
| 75 | + Wrapper kWrap = WrapUtil.getKeyWrapper(spec, secret); |
| 76 | + secretKey = kWrap.wrap(secretKey, 0, secretKey.length); |
| 77 | +// secretKey = Arrays.concatenate(encapsulation, kWrap.wrap(secretKey, 0, secretKey.length)); |
| 78 | + } |
| 79 | + catch (InvalidKeyException e) |
| 80 | + { |
| 81 | + throw new RuntimeException(e); |
| 82 | + } |
71 | 83 | }
|
| 84 | + |
| 85 | + return new KEM.Encapsulated(new SecretKeySpec(secretKey, algorithm), encapsulation, parameterSpec.getOtherInfo()); |
| 86 | + |
72 | 87 | }
|
73 | 88 |
|
74 | 89 | @Override
|
75 | 90 | public int engineSecretSize()
|
76 | 91 | {
|
77 |
| - return publicKey.getKeyParams().getParameters().getSessionKeySize() / 8; |
| 92 | + return parameterSpec.getKeySize() / 8; |
78 | 93 | }
|
79 | 94 |
|
80 | 95 | @Override
|
|
0 commit comments