Skip to content
Open
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -1,9 +1,8 @@
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.

using System.Collections.Generic;
using System.Diagnostics;
using System.Linq;
using System.Formats.Asn1;
using System.Numerics;
using Test.Cryptography;

namespace System.Security.Cryptography.X509Certificates.Tests.CertificateCreation
Expand Down Expand Up @@ -35,109 +34,9 @@ public override byte[] GetSignatureAlgorithmIdentifier(HashAlgorithmName hashAlg
throw new InvalidOperationException();
}

private static byte[] EncodeLength(int length)
{
Debug.Assert(length >= 0);

byte low = unchecked((byte)length);

// If the length value fits in 7 bits, it's an answer all by itself.
if (length < 0x80)
{
return new[] { low };
}

if (length <= 0xFF)
{
return new byte[] { 0x81, low };
}

int remainder = length >> 8;
byte midLow = unchecked((byte)remainder);

if (length <= 0xFFFF)
{
return new byte[] { 0x82, midLow, low };
}

remainder >>= 8;
byte midHigh = unchecked((byte)remainder);

if (length <= 0xFFFFFF)
{
return new byte[] { 0x83, midHigh, midLow, low };
}

remainder >>= 8;
byte high = unchecked((byte)remainder);

// Since we know this was a non-negative signed number, the highest
// legal value here is 0x7F.
Debug.Assert(remainder < 0x80);

return new byte[] { 0x84, high, midHigh, midLow, low };
}

private byte[] EncodeUnsignedInteger(byte[] data)
{
return EncodeUnsignedInteger(data, 0, data.Length);
}

private byte[] EncodeUnsignedInteger(byte[] data, int offset, int count)
{
int length = count;
int realOffset = offset;
bool paddingByte = false;

if (count == 0 || data[offset] >= 0x80)
{
paddingByte = true;
}
else
{
while (length > 1 && data[realOffset] == 0)
{
realOffset++;
length--;
}
}

byte encodedLength = (byte)length;

if (paddingByte)
{
encodedLength++;
}

IEnumerable<byte> bytes = new byte[] { 0x02 };
bytes = bytes.Concat(EncodeLength(encodedLength));

if (paddingByte)
{
bytes = bytes.Concat(new byte[1]);
}

bytes = bytes.Concat(data.Skip(realOffset).Take(length));

return bytes.ToArray();
}

public override byte[] SignData(byte[] data, HashAlgorithmName hashAlgorithm)
{
byte[] ieeeFormat = _key.SignData(data, hashAlgorithm);

Debug.Assert(ieeeFormat.Length % 2 == 0);
int segmentLength = ieeeFormat.Length / 2;

byte[] r = EncodeUnsignedInteger(ieeeFormat, 0, segmentLength);
byte[] s = EncodeUnsignedInteger(ieeeFormat, segmentLength, segmentLength);

return
new byte[] { 0x30 }.
Concat(EncodeLength(r.Length + s.Length)).
Concat(r).
Concat(s).
ToArray();
return _key.SignData(data, hashAlgorithm, DSASignatureFormat.Rfc3279DerSequence);
}

protected override PublicKey BuildPublicKey()
Expand All @@ -153,24 +52,31 @@ protected override PublicKey BuildPublicKey()
// g INTEGER
// }

byte[] p = EncodeUnsignedInteger(dsaParameters.P);
byte[] q = EncodeUnsignedInteger(dsaParameters.Q);
byte[] g = EncodeUnsignedInteger(dsaParameters.G);
AsnWriter writer = new AsnWriter(AsnEncodingRules.DER);

byte[] algParameters =
new byte[] { 0x30 }.
Concat(EncodeLength(p.Length + q.Length + g.Length)).
Concat(p).
Concat(q).
Concat(g).
ToArray();
using (writer.PushSequence())
{
WriteUnsignedInteger(writer, dsaParameters.P);
WriteUnsignedInteger(writer, dsaParameters.Q);
WriteUnsignedInteger(writer, dsaParameters.G);
}

byte[] algParameters = writer.Encode();

byte[] keyValue = EncodeUnsignedInteger(dsaParameters.Y);
writer.Reset();
WriteUnsignedInteger(writer, dsaParameters.Y);
byte[] keyValue = writer.Encode();

return new PublicKey(
oid,
new AsnEncodedData(oid, algParameters),
new AsnEncodedData(oid, keyValue));
}

private static void WriteUnsignedInteger(AsnWriter writer, ReadOnlySpan<byte> value)
{
BigInteger integer = new BigInteger(value, isUnsigned: true, isBigEndian: true);
writer.WriteInteger(integer);
}
}
}
Loading