summary refs log tree commit diff
path: root/crypto
diff options
context:
space:
mode:
authorPeter Dettman <peter.dettman@bouncycastle.org>2023-06-14 14:04:48 +0700
committerPeter Dettman <peter.dettman@bouncycastle.org>2023-06-14 14:04:48 +0700
commit75cb55fedb8b90dc99996b88ccb55a35549be184 (patch)
tree1d9c98fa82bb48cc5cf704c6623e5e407a90671e /crypto
parentAdd new variant of GetBasicAgreementWithKdf (diff)
downloadBouncyCastle.NET-ed25519-75cb55fedb8b90dc99996b88ccb55a35549be184.tar.xz
Add ECDHCWithKdfBasicAgreement
- refactor other WithKdf agreeements
- support "ECCDHWITHSHA1KDF" in AgreementUtilities
Diffstat (limited to 'crypto')
-rw-r--r--crypto/src/crypto/agreement/BasicAgreementWithKdf.cs34
-rw-r--r--crypto/src/crypto/agreement/ECDHBasicAgreement.cs5
-rw-r--r--crypto/src/crypto/agreement/ECDHCBasicAgreement.cs4
-rw-r--r--crypto/src/crypto/agreement/ECDHCWithKdfBasicAgreement.cs30
-rw-r--r--crypto/src/crypto/agreement/ECDHWithKdfBasicAgreement.cs48
-rw-r--r--crypto/src/crypto/agreement/ECMqvBasicAgreement.cs10
-rw-r--r--crypto/src/crypto/agreement/ECMqvWithKdfBasicAgreement.cs50
-rw-r--r--crypto/src/security/AgreementUtilities.cs17
8 files changed, 103 insertions, 95 deletions
diff --git a/crypto/src/crypto/agreement/BasicAgreementWithKdf.cs b/crypto/src/crypto/agreement/BasicAgreementWithKdf.cs
new file mode 100644
index 000000000..8f78bd393
--- /dev/null
+++ b/crypto/src/crypto/agreement/BasicAgreementWithKdf.cs
@@ -0,0 +1,34 @@
+using System.Security.Cryptography;
+
+using Org.BouncyCastle.Asn1;
+using Org.BouncyCastle.Crypto.Agreement.Kdf;
+using Org.BouncyCastle.Math;
+using Org.BouncyCastle.Security;
+using Org.BouncyCastle.Utilities;
+
+namespace Org.BouncyCastle.Crypto.Agreement
+{
+    internal static class BasicAgreementWithKdf
+    {
+        internal static BigInteger CalculateAgreementWithKdf(string algorithm, IDerivationFunction kdf, int fieldSize,
+            BigInteger result)
+        {
+            // Note that the ec.KeyAgreement class in JCE only uses kdf in one
+            // of the engineGenerateSecret methods.
+
+            int keySize = GeneratorUtilities.GetDefaultKeySize(algorithm);
+
+            DHKdfParameters dhKdfParams = new DHKdfParameters(
+                new DerObjectIdentifier(algorithm),
+                keySize,
+                BigIntegers.AsUnsignedByteArray(fieldSize, result));
+
+            kdf.Init(dhKdfParams);
+
+            byte[] keyBytes = new byte[keySize / 8];
+            kdf.GenerateBytes(keyBytes, 0, keyBytes.Length);
+
+            return new BigInteger(1, keyBytes);
+        }
+    }
+}
diff --git a/crypto/src/crypto/agreement/ECDHBasicAgreement.cs b/crypto/src/crypto/agreement/ECDHBasicAgreement.cs
index 4555cdde4..700bb88e6 100644
--- a/crypto/src/crypto/agreement/ECDHBasicAgreement.cs
+++ b/crypto/src/crypto/agreement/ECDHBasicAgreement.cs
@@ -21,9 +21,11 @@ namespace Org.BouncyCastle.Crypto.Agreement
      * DL/ECKAS-DH2. It assumes that the input keys are valid (see also
      * Section 7.2.2).
      */
+    // TODO[api] sealed
     public class ECDHBasicAgreement
         : IBasicAgreement
     {
+        // TODO[api] private
         protected internal ECPrivateKeyParameters privKey;
 
         public virtual void Init(ICipherParameters parameters)
@@ -44,8 +46,7 @@ namespace Org.BouncyCastle.Crypto.Agreement
             return (privKey.Parameters.Curve.FieldSize + 7) / 8;
         }
 
-        public virtual BigInteger CalculateAgreement(
-            ICipherParameters pubKey)
+        public virtual BigInteger CalculateAgreement(ICipherParameters pubKey)
         {
             ECPublicKeyParameters pub = (ECPublicKeyParameters)pubKey;
             ECDomainParameters dp = privKey.Parameters;
diff --git a/crypto/src/crypto/agreement/ECDHCBasicAgreement.cs b/crypto/src/crypto/agreement/ECDHCBasicAgreement.cs
index bb4c185df..a58974953 100644
--- a/crypto/src/crypto/agreement/ECDHCBasicAgreement.cs
+++ b/crypto/src/crypto/agreement/ECDHCBasicAgreement.cs
@@ -26,6 +26,7 @@ namespace Org.BouncyCastle.Crypto.Agreement
      * (if you want that just use ECDHBasicAgreement and note they both implement
      * BasicAgreement!).</p>
      */
+    // TODO[api] sealed
     public class ECDHCBasicAgreement
         : IBasicAgreement
     {
@@ -49,8 +50,7 @@ namespace Org.BouncyCastle.Crypto.Agreement
             return (privKey.Parameters.Curve.FieldSize + 7) / 8;
         }
 
-        public virtual BigInteger CalculateAgreement(
-            ICipherParameters pubKey)
+        public virtual BigInteger CalculateAgreement(ICipherParameters pubKey)
         {
             ECPublicKeyParameters pub = (ECPublicKeyParameters)pubKey;
             ECDomainParameters dp = privKey.Parameters;
diff --git a/crypto/src/crypto/agreement/ECDHCWithKdfBasicAgreement.cs b/crypto/src/crypto/agreement/ECDHCWithKdfBasicAgreement.cs
new file mode 100644
index 000000000..0f38c0fe6
--- /dev/null
+++ b/crypto/src/crypto/agreement/ECDHCWithKdfBasicAgreement.cs
@@ -0,0 +1,30 @@
+using System;
+
+using Org.BouncyCastle.Asn1;
+using Org.BouncyCastle.Crypto.Agreement.Kdf;
+using Org.BouncyCastle.Math;
+using Org.BouncyCastle.Security;
+using Org.BouncyCastle.Utilities;
+
+namespace Org.BouncyCastle.Crypto.Agreement
+{
+    public sealed class ECDHCWithKdfBasicAgreement
+        : ECDHCBasicAgreement
+    {
+        private readonly string m_algorithm;
+        private readonly IDerivationFunction m_kdf;
+
+        public ECDHCWithKdfBasicAgreement(string algorithm, IDerivationFunction kdf)
+        {
+            m_algorithm = algorithm ?? throw new ArgumentNullException(nameof(algorithm));
+            m_kdf = kdf ?? throw new ArgumentNullException(nameof(kdf));
+        }
+
+        public override BigInteger CalculateAgreement(ICipherParameters pubKey)
+        {
+            BigInteger result = base.CalculateAgreement(pubKey);
+
+            return BasicAgreementWithKdf.CalculateAgreementWithKdf(m_algorithm, m_kdf, GetFieldSize(), result);
+        }
+    }
+}
diff --git a/crypto/src/crypto/agreement/ECDHWithKdfBasicAgreement.cs b/crypto/src/crypto/agreement/ECDHWithKdfBasicAgreement.cs
index a73e3f438..fb85da611 100644
--- a/crypto/src/crypto/agreement/ECDHWithKdfBasicAgreement.cs
+++ b/crypto/src/crypto/agreement/ECDHWithKdfBasicAgreement.cs
@@ -1,59 +1,31 @@
 using System;
 
 using Org.BouncyCastle.Asn1;
-using Org.BouncyCastle.Asn1.X9;
 using Org.BouncyCastle.Crypto.Agreement.Kdf;
 using Org.BouncyCastle.Math;
 using Org.BouncyCastle.Security;
+using Org.BouncyCastle.Utilities;
 
 namespace Org.BouncyCastle.Crypto.Agreement
 {
-	public class ECDHWithKdfBasicAgreement
+    // TODO[api] sealed
+    public class ECDHWithKdfBasicAgreement
 		: ECDHBasicAgreement
 	{
-		private readonly string algorithm;
-		private readonly IDerivationFunction kdf;
+		private readonly string m_algorithm;
+		private readonly IDerivationFunction m_kdf;
 
-		public ECDHWithKdfBasicAgreement(
-			string				algorithm,
-			IDerivationFunction	kdf)
+		public ECDHWithKdfBasicAgreement(string algorithm, IDerivationFunction kdf)
 		{
-			if (algorithm == null)
-				throw new ArgumentNullException("algorithm");
-			if (kdf == null)
-				throw new ArgumentNullException("kdf");
-
-			this.algorithm = algorithm;
-			this.kdf = kdf;
+			m_algorithm = algorithm ?? throw new ArgumentNullException(nameof(algorithm));
+			m_kdf = kdf ?? throw new ArgumentNullException(nameof(kdf));
 		}
 
-		public override BigInteger CalculateAgreement(
-			ICipherParameters pubKey)
+		public override BigInteger CalculateAgreement(ICipherParameters pubKey)
 		{
-			// Note that the ec.KeyAgreement class in JCE only uses kdf in one
-			// of the engineGenerateSecret methods.
-
 			BigInteger result = base.CalculateAgreement(pubKey);
 
-			int keySize = GeneratorUtilities.GetDefaultKeySize(algorithm);
-
-			DHKdfParameters dhKdfParams = new DHKdfParameters(
-				new DerObjectIdentifier(algorithm),
-				keySize,
-				BigIntToBytes(result));
-
-			kdf.Init(dhKdfParams);
-
-			byte[] keyBytes = new byte[keySize / 8];
-			kdf.GenerateBytes(keyBytes, 0, keyBytes.Length);
-
-			return new BigInteger(1, keyBytes);
-		}
-
-		private byte[] BigIntToBytes(BigInteger r)
-		{
-			int byteLength = X9IntegerConverter.GetByteLength(privKey.Parameters.Curve);
-			return X9IntegerConverter.IntegerToBytes(r, byteLength);
+			return BasicAgreementWithKdf.CalculateAgreementWithKdf(m_algorithm, m_kdf, GetFieldSize(), result);
 		}
 	}
 }
diff --git a/crypto/src/crypto/agreement/ECMqvBasicAgreement.cs b/crypto/src/crypto/agreement/ECMqvBasicAgreement.cs
index 984d66587..64f41dd78 100644
--- a/crypto/src/crypto/agreement/ECMqvBasicAgreement.cs
+++ b/crypto/src/crypto/agreement/ECMqvBasicAgreement.cs
@@ -7,9 +7,11 @@ using Org.BouncyCastle.Math.EC;
 
 namespace Org.BouncyCastle.Crypto.Agreement
 {
+    // TODO[api] sealed
     public class ECMqvBasicAgreement
         : IBasicAgreement
     {
+        // TODO[api] private
         protected internal MqvPrivateParameters privParams;
 
         public virtual void Init(ICipherParameters parameters)
@@ -19,7 +21,10 @@ namespace Org.BouncyCastle.Crypto.Agreement
                 parameters = withRandom.Parameters;
             }
 
-            this.privParams = (MqvPrivateParameters)parameters;
+            if (!(parameters is MqvPrivateParameters mqvPrivateParameters))
+                throw new ArgumentException("ECMqvBasicAgreement expects MqvPrivateParameters");
+
+            this.privParams = mqvPrivateParameters;
         }
 
         public virtual int GetFieldSize()
@@ -27,8 +32,7 @@ namespace Org.BouncyCastle.Crypto.Agreement
             return (privParams.StaticPrivateKey.Parameters.Curve.FieldSize + 7) / 8;
         }
 
-        public virtual BigInteger CalculateAgreement(
-            ICipherParameters pubKey)
+        public virtual BigInteger CalculateAgreement(ICipherParameters pubKey)
         {
             MqvPublicParameters pubParams = (MqvPublicParameters)pubKey;
 
diff --git a/crypto/src/crypto/agreement/ECMqvWithKdfBasicAgreement.cs b/crypto/src/crypto/agreement/ECMqvWithKdfBasicAgreement.cs
index 248e04e27..6977a5057 100644
--- a/crypto/src/crypto/agreement/ECMqvWithKdfBasicAgreement.cs
+++ b/crypto/src/crypto/agreement/ECMqvWithKdfBasicAgreement.cs
@@ -1,59 +1,31 @@
 using System;
 
 using Org.BouncyCastle.Asn1;
-using Org.BouncyCastle.Asn1.X9;
 using Org.BouncyCastle.Crypto.Agreement.Kdf;
 using Org.BouncyCastle.Math;
 using Org.BouncyCastle.Security;
+using Org.BouncyCastle.Utilities;
 
 namespace Org.BouncyCastle.Crypto.Agreement
 {
-	public class ECMqvWithKdfBasicAgreement
+    // TODO[api] sealed
+    public class ECMqvWithKdfBasicAgreement
 		: ECMqvBasicAgreement
 	{
-		private readonly string algorithm;
-		private readonly IDerivationFunction kdf;
+		private readonly string m_algorithm;
+		private readonly IDerivationFunction m_kdf;
 
-		public ECMqvWithKdfBasicAgreement(
-			string				algorithm,
-			IDerivationFunction	kdf)
+		public ECMqvWithKdfBasicAgreement(string algorithm, IDerivationFunction kdf)
 		{
-			if (algorithm == null)
-				throw new ArgumentNullException("algorithm");
-			if (kdf == null)
-				throw new ArgumentNullException("kdf");
-
-			this.algorithm = algorithm;
-			this.kdf = kdf;
+            m_algorithm = algorithm ?? throw new ArgumentNullException(nameof(algorithm));
+            m_kdf = kdf ?? throw new ArgumentNullException(nameof(kdf));
 		}
 
-		public override BigInteger CalculateAgreement(
-			ICipherParameters pubKey)
+		public override BigInteger CalculateAgreement(ICipherParameters pubKey)
 		{
-			// Note that the ec.KeyAgreement class in JCE only uses kdf in one
-			// of the engineGenerateSecret methods.
-
-			BigInteger result = base.CalculateAgreement(pubKey);
-
-			int keySize = GeneratorUtilities.GetDefaultKeySize(algorithm);
-
-			DHKdfParameters dhKdfParams = new DHKdfParameters(
-				new DerObjectIdentifier(algorithm),
-				keySize,
-				BigIntToBytes(result));
-
-			kdf.Init(dhKdfParams);
+            BigInteger result = base.CalculateAgreement(pubKey);
 
-			byte[] keyBytes = new byte[keySize / 8];
-			kdf.GenerateBytes(keyBytes, 0, keyBytes.Length);
-
-			return new BigInteger(1, keyBytes);
-		}
-
-		private byte[] BigIntToBytes(BigInteger r)
-		{
-			int byteLength = X9IntegerConverter.GetByteLength(privParams.StaticPrivateKey.Parameters.Curve);
-			return X9IntegerConverter.IntegerToBytes(r, byteLength);
+            return BasicAgreementWithKdf.CalculateAgreementWithKdf(m_algorithm, m_kdf, GetFieldSize(), result);
 		}
 	}
 }
diff --git a/crypto/src/security/AgreementUtilities.cs b/crypto/src/security/AgreementUtilities.cs
index 5fef5f877..041aeeed2 100644
--- a/crypto/src/security/AgreementUtilities.cs
+++ b/crypto/src/security/AgreementUtilities.cs
@@ -67,24 +67,19 @@ namespace Org.BouncyCastle.Security
 			return GetBasicAgreementWithKdf(oid.Id, wrapAlgorithm);
 		}
 
-		public static IBasicAgreement GetBasicAgreementWithKdf(
-			string agreeAlgorithm,
-			string wrapAlgorithm)
+		public static IBasicAgreement GetBasicAgreementWithKdf(string agreeAlgorithm, string wrapAlgorithm)
 		{
             string mechanism = GetMechanism(agreeAlgorithm);
 
             // 'DHWITHSHA1KDF' retained for backward compatibility
 			if (mechanism == "DHWITHSHA1KDF" || mechanism == "ECDHWITHSHA1KDF")
-				return new ECDHWithKdfBasicAgreement(
-					wrapAlgorithm,
-					new ECDHKekGenerator(
-						new Sha1Digest()));
+				return new ECDHWithKdfBasicAgreement(wrapAlgorithm, new ECDHKekGenerator(new Sha1Digest()));
+
+			if (mechanism == "ECCDHWITHSHA1KDF")
+				return new ECDHCWithKdfBasicAgreement(wrapAlgorithm, new ECDHKekGenerator(new Sha1Digest()));
 
 			if (mechanism == "ECMQVWITHSHA1KDF")
-				return new ECMqvWithKdfBasicAgreement(
-					wrapAlgorithm,
-					new ECDHKekGenerator(
-						new Sha1Digest()));
+				return new ECMqvWithKdfBasicAgreement(wrapAlgorithm, new ECDHKekGenerator(new Sha1Digest()));
 
 			throw new SecurityUtilityException("Basic Agreement (with KDF) " + agreeAlgorithm + " not recognised.");
 		}