summary refs log tree commit diff
path: root/crypto/src/asn1/Asn1RelativeOid.cs
diff options
context:
space:
mode:
authorPeter Dettman <peter.dettman@bouncycastle.org>2024-03-08 19:09:32 +0700
committerPeter Dettman <peter.dettman@bouncycastle.org>2024-03-08 19:09:32 +0700
commite915a3a8e4c11a54210d8df91e3c2386f968376b (patch)
tree77cf3bb8f561a93ade91365ed041b93bbefa01e3 /crypto/src/asn1/Asn1RelativeOid.cs
parentRefactoring (diff)
downloadBouncyCastle.NET-ed25519-e915a3a8e4c11a54210d8df91e3c2386f968376b.tar.xz
Add Asn1RelativeOid cache
Diffstat (limited to 'crypto/src/asn1/Asn1RelativeOid.cs')
-rw-r--r--crypto/src/asn1/Asn1RelativeOid.cs25
1 files changed, 24 insertions, 1 deletions
diff --git a/crypto/src/asn1/Asn1RelativeOid.cs b/crypto/src/asn1/Asn1RelativeOid.cs
index f43a85479..3e2b54c9e 100644
--- a/crypto/src/asn1/Asn1RelativeOid.cs
+++ b/crypto/src/asn1/Asn1RelativeOid.cs
@@ -1,6 +1,7 @@
 using System;
 using System.IO;
 using System.Text;
+using System.Threading;
 
 using Org.BouncyCastle.Math;
 using Org.BouncyCastle.Utilities;
@@ -80,6 +81,8 @@ namespace Org.BouncyCastle.Asn1
 
         private const long LongLimit = (long.MaxValue >> 7) - 0x7F;
 
+        private static readonly Asn1RelativeOid[] Cache = new Asn1RelativeOid[64];
+
         private readonly byte[] m_contents;
         private string m_identifier;
 
@@ -167,7 +170,27 @@ namespace Org.BouncyCastle.Asn1
 
         internal static Asn1RelativeOid CreatePrimitive(byte[] contents, bool clone)
         {
-            return new Asn1RelativeOid(contents, clone);
+            uint index = (uint)Arrays.GetHashCode(contents);
+
+            index ^= index >> 24;
+            index ^= index >> 12;
+            index ^= index >> 6;
+            index &= 63;
+
+            var originalEntry = Volatile.Read(ref Cache[index]);
+            if (originalEntry != null && Arrays.AreEqual(contents, originalEntry.m_contents))
+                return originalEntry;
+
+            var newEntry = new Asn1RelativeOid(contents, clone);
+
+            var exchangedEntry = Interlocked.CompareExchange(ref Cache[index], newEntry, originalEntry);
+            if (exchangedEntry != originalEntry)
+            {
+                if (exchangedEntry != null && Arrays.AreEqual(contents, exchangedEntry.m_contents))
+                    return exchangedEntry;
+            }
+
+            return newEntry;
         }
 
         internal static bool IsValidContents(byte[] contents)