You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			188 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			188 lines
		
	
	
		
			6.5 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //------------------------------------------------------------ | |||
|  | // Copyright (c) Microsoft Corporation.  All rights reserved. | |||
|  | //------------------------------------------------------------ | |||
|  | 
 | |||
|  | namespace System.IdentityModel | |||
|  | { | |||
|  |     using System.Collections.Generic; | |||
|  |     using System.Text; | |||
|  | 
 | |||
|  |     static class Asn1IntegerConverter | |||
|  |     { | |||
|  |         static List<byte[]> powersOfTwo = new List<byte[]>(new byte[][] { new byte[] { 1 } }); | |||
|  |         readonly static char[] digitMap = new char[] { '0', '1', '2', '3', '4', '5', '6', '7', '8', '9' }; | |||
|  | 
 | |||
|  |         public static string Asn1IntegerToDecimalString(byte[] asn1) | |||
|  |         { | |||
|  |             if (asn1 == null) | |||
|  |                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperArgumentNull("asn1"); | |||
|  | 
 | |||
|  |             if (asn1.Length == 0) | |||
|  |                 throw DiagnosticUtility.ExceptionUtility.ThrowHelperError(new ArgumentOutOfRangeException("asn1", SR.GetString(SR.LengthOfArrayToConvertMustGreaterThanZero))); | |||
|  | 
 | |||
|  | 
 | |||
|  |             List<byte> positiveDecimalDigits = new List<byte>((asn1.Length * 8) / 3); | |||
|  |             int absoluteBitNumber = 0; | |||
|  |             byte currentByte; | |||
|  | 
 | |||
|  |             // Since X509Certificate.GetSerialNumber return the little-endian,  | |||
|  |             // the most significant is at the last byte. | |||
|  |             for (int byteNumber = 0; byteNumber < asn1.Length - 1; byteNumber++) | |||
|  |             { | |||
|  |                 currentByte = asn1[byteNumber]; | |||
|  |                 for (int i = 0; i < 8; i++) | |||
|  |                 { | |||
|  |                     if ((currentByte & 1) == 1) | |||
|  |                     { | |||
|  |                         AddSecondDecimalToFirst(positiveDecimalDigits, TwoToThePowerOf(absoluteBitNumber)); | |||
|  |                     } | |||
|  |                     absoluteBitNumber++; | |||
|  |                     currentByte >>= 1; | |||
|  |                 } | |||
|  |             } | |||
|  | 
 | |||
|  |             // Special case the most significant bit of the most significant byte as a negative value | |||
|  |             currentByte = asn1[asn1.Length - 1]; | |||
|  |             for (int i = 0; i < 7; i++) | |||
|  |             { | |||
|  |                 if ((currentByte & 1) == 1) | |||
|  |                 { | |||
|  |                     AddSecondDecimalToFirst(positiveDecimalDigits, TwoToThePowerOf(absoluteBitNumber)); | |||
|  |                 } | |||
|  |                 absoluteBitNumber++; | |||
|  |                 currentByte >>= 1; | |||
|  |             } | |||
|  | 
 | |||
|  |             StringBuilder result = new StringBuilder(positiveDecimalDigits.Count + 1); | |||
|  |             List<byte> resultDigits = null; | |||
|  |             if (currentByte == 0) | |||
|  |             { | |||
|  |                 // positive number | |||
|  |                 resultDigits = positiveDecimalDigits; | |||
|  |             } | |||
|  |             else | |||
|  |             { | |||
|  |                 // negative number | |||
|  |                 List<byte> negativeDecimalDigits = new List<byte>(TwoToThePowerOf(absoluteBitNumber)); | |||
|  |                 SubtractSecondDecimalFromFirst(negativeDecimalDigits, positiveDecimalDigits); | |||
|  |                 resultDigits = negativeDecimalDigits; | |||
|  |                 result.Append('-'); | |||
|  |             } | |||
|  |             int d; | |||
|  |             for (d = resultDigits.Count - 1; d >= 0; d--) | |||
|  |             { | |||
|  |                 if (resultDigits[d] != 0) | |||
|  |                     break; | |||
|  |             } | |||
|  | 
 | |||
|  |             if (d < 0 && asn1.Length > 0) | |||
|  |             { | |||
|  |                 // This is a special case where the result contains 0 | |||
|  |                 result.Append(digitMap[0]); | |||
|  |             } | |||
|  |             else | |||
|  |             { | |||
|  |                 while (d >= 0) | |||
|  |                 { | |||
|  |                     result.Append(digitMap[resultDigits[d--]]); | |||
|  |                 } | |||
|  |             } | |||
|  | 
 | |||
|  |             return result.ToString(); | |||
|  |         } | |||
|  | 
 | |||
|  |         static byte[] TwoToThePowerOf(int n) | |||
|  |         { | |||
|  |             lock (powersOfTwo) | |||
|  |             { | |||
|  |                 if (n >= powersOfTwo.Count) | |||
|  |                 { | |||
|  |                     for (int power = powersOfTwo.Count; power <= n; power++) | |||
|  |                     { | |||
|  |                         List<byte> decimalDigits = new List<byte>(powersOfTwo[power - 1]); | |||
|  |                         byte carryover = 0; | |||
|  |                         for (int i = 0; i < decimalDigits.Count; i++) | |||
|  |                         { | |||
|  |                             byte newValue = (byte)((decimalDigits[i] << 1) + carryover); | |||
|  |                             decimalDigits[i] = (byte)(newValue % 10); | |||
|  |                             carryover = (byte)(newValue / 10); | |||
|  |                         } | |||
|  |                         if (carryover > 0) | |||
|  |                         { | |||
|  |                             decimalDigits.Add(carryover); | |||
|  |                             carryover = 0; | |||
|  |                         } | |||
|  |                         powersOfTwo.Add(decimalDigits.ToArray()); | |||
|  |                     } | |||
|  |                 } | |||
|  |                 return powersOfTwo[n]; | |||
|  |             } | |||
|  |         } | |||
|  | 
 | |||
|  |         static void AddSecondDecimalToFirst(List<byte> first, byte[] second) | |||
|  |         { | |||
|  |             byte carryover = 0; | |||
|  |             for (int i = 0; i < second.Length || i < first.Count; i++) | |||
|  |             { | |||
|  |                 if (i >= first.Count) | |||
|  |                 { | |||
|  |                     first.Add(0); | |||
|  |                 } | |||
|  |                 byte newValue; | |||
|  |                 if (i < second.Length) | |||
|  |                 { | |||
|  |                     newValue = (byte)(first[i] + second[i] + carryover); | |||
|  |                 } | |||
|  |                 else | |||
|  |                 { | |||
|  |                     newValue = (byte)(first[i] + carryover); | |||
|  |                 } | |||
|  |                 first[i] = (byte)(newValue % 10); | |||
|  |                 carryover = (byte)(newValue / 10); | |||
|  |             } | |||
|  |             if (carryover > 0) | |||
|  |             { | |||
|  |                 first.Add(carryover); | |||
|  |             } | |||
|  |         } | |||
|  | 
 | |||
|  |         static void SubtractSecondDecimalFromFirst(List<byte> first, List<byte> second) | |||
|  |         { | |||
|  |             byte borrow = 0; | |||
|  |             for (int i = 0; i < second.Count; i++) | |||
|  |             { | |||
|  |                 int newValue = first[i] - second[i] - borrow; | |||
|  |                 if (newValue < 0) | |||
|  |                 { | |||
|  |                     borrow = 1; | |||
|  |                     first[i] = (byte)(newValue + 10); | |||
|  |                 } | |||
|  |                 else | |||
|  |                 { | |||
|  |                     borrow = 0; | |||
|  |                     first[i] = (byte)newValue; | |||
|  |                 } | |||
|  |             } | |||
|  |             if (borrow > 0) | |||
|  |             { | |||
|  |                 for (int i = second.Count; i < first.Count; i++) | |||
|  |                 { | |||
|  |                     int newValue = first[i] - borrow; | |||
|  |                     if (newValue < 0) | |||
|  |                     { | |||
|  |                         borrow = 1; | |||
|  |                         first[i] = (byte)(newValue + 10); | |||
|  |                     } | |||
|  |                     else | |||
|  |                     { | |||
|  |                         borrow = 0; | |||
|  |                         first[i] = (byte)newValue; | |||
|  |                         break; | |||
|  |                     } | |||
|  |                 } | |||
|  |             } | |||
|  |             DiagnosticUtility.DebugAssert(borrow == 0, ""); | |||
|  |         } | |||
|  |     } | |||
|  | } |