VMProtect/utils/ipn_sqlclr/keygen.cs
VNGhostMans 5ec92ee05e first commit
Version 3.x.x
2023-05-14 20:21:09 +07:00

213 lines
7.3 KiB
C#

using System;
using System.IO;
using System.Numerics;
using System.Security.Cryptography;
using System.Text;
namespace ipn_sqlclr
{
enum SerialNumberChunks : byte
{
Version = 0x01, // 1 byte of data - version
UserName = 0x02, // 1 + N bytes - length + N bytes of customer's name (without enging \0).
Email = 0x03, // 1 + N bytes - length + N bytes of customer's email (without ending \0).
ProductCode = 0x07, // 8 bytes - used for decrypting some parts of exe-file
UserData = 0x08, // 1 + N bytes - length + N bytes of user data
MaxBuild = 0x09, // 4 bytes - (year << 16) + (month << 8) + (day)
End = 0xFF // 4 bytes - checksum: the first four bytes of sha-1 hash from the data before that chunk
};
public static class Rsa
{
private const string PublicExpB64 = "AAEAAQ==";
private const string PrivateExpB64 = "CXHXWx/Z9JqetQWwFpvmD72wrDiqQOXMQs18fhAMjWCfJ/f2r3p2io+iB3gqIuu3LGH3WJ8PQuIzvDMnbwAx+8BbAyYhWhGEbxDdifndjQ2KlDV2Hu8NQgCbc5Wjok0rKwQ+Bxeb2i1+Gu3FsnhRNv9RhSyiwcnH/4Q3+ySE3AFAcAUwuQABePjDKCYOfIyx7RKz5h0sG+v10nkPuuCGPSnh+AXDTBIJFH+yNIjkrfweC9A3dv7URyRJumAMgm/SnDU76rTkFw9vZpupQeMtMtIsZIkeFSngip9KImD5zzbb2vKD63Cg9W/Yvqgvro/d+cR5n6P0t4DzfanNIFRGpFrX8/Q5VjuezDKw/4YbsFYwOhzJPRxglmCEjh8cpfxJ11cUXa/hNBV4c4Dp29D0F+w01OlBnFb1Ck9VXur2qJCsqcWtjsnt/VITsxa1jzr+3C2+uvaI4JSd7yLEnTqSaSsRfWuhDXgjY/YWhmyvMzeQeXBGOXKt2j2lY2Fm0WJx";
private const string ModulusB64 = "pwUqwaM8IOukyx06Lvi5YNQ70JE7pwg7K+pmM/vCe1CUseHKFM1v1m11geDjVsAt38AnaiFs3JhtTs80ySCIxOSyvMw6Cd52k6N6dn7LAx1mxQLJLhYeMMJYbplMHnMLwYN0+IO58OVbEqRyaJV2ExolnK2EYZL7QRXujGY7/sOoOMF3p6GsWJK6kkBJICIoL9hHWBQMO6/9rmls/+EhaWuP80Vx0+H2OlrQ58K+TJeyE393cvb4QufiEPpCNaB50Klee9QUnsjSW/bTnmGn4Bi5+cowRbawUY73Q5I58fMAXiH9ueDPuNMR9YKDgW9GxunLmYkbuwqIp/v7kw3cfMBM0ihhB0B8UhjyAMAGLzJWX3H/H6Zrz41g9PbPjTAxfsTaCrxoqjaTaO4zk9YsI//VX9Fhivcy913SevBpNandziGfYH/oHW2xDy9AfwkE1wuIBlLj7c/k8U1YmmRAmkoCzlmB7EU4ClNltboh1uARUQ6wW30upppnuYhGkTy7";
static BigInteger B2Bi(byte[] b) //reverse & make positive
{
Array.Reverse(b);
var b2 = new byte[b.Length + 1];
Array.Copy(b, b2, b.Length);
return new BigInteger(b2);
}
private static readonly BigInteger PublicExp = B2Bi(Convert.FromBase64String(PublicExpB64));
private static readonly BigInteger PrivateExp = B2Bi(Convert.FromBase64String(PrivateExpB64));
private static readonly BigInteger Modulus = B2Bi(Convert.FromBase64String(ModulusB64));
public static byte[] Encrypt(byte[] paddedData)
{
var x = B2Bi(paddedData);
var y = BigInteger.ModPow(x, PrivateExp, Modulus);
byte[] ret = y.ToByteArray();
Array.Resize(ref ret, paddedData.Length);
Array.Reverse(ret);
return ret;
}
public static byte[] Decrypt(byte[] data)
{
var x = B2Bi(data);
var y = BigInteger.ModPow(x, PublicExp, Modulus);
byte[] ret = y.ToByteArray();
Array.Reverse(ret);
return ret;
}
}
public static class Keygen
{
public static void ParseKey(string key, out int productId, out string customerName, out string eMail, out DateTime maxBuildDt)
{
productId = -1;
customerName = null;
eMail = null;
maxBuildDt = new DateTime();
var crypted = Convert.FromBase64String(key);
var data = Rsa.Decrypt(crypted);
int i;
for (i = 2; i < data.Length && data[i] != 0; i++) {
}
i++;
var pos = i;
while (pos < data.Length)
{
var b = data[pos++];
switch (b)
{
case (byte) SerialNumberChunks.Version:
b = data[pos++];
if (b < 1 || b > 2)
throw new InvalidDataException("SerialNumberChunks.Version");
break;
case (byte) SerialNumberChunks.UserName:
b = data[pos++];
customerName = Encoding.UTF8.GetString(data, pos, b);
pos += b;
break;
case (byte) SerialNumberChunks.Email:
b = data[pos++];
eMail = Encoding.UTF8.GetString(data, pos, b);
pos += b;
break;
case (byte)SerialNumberChunks.ProductCode:
pos += 8;
break;
case (byte) SerialNumberChunks.UserData:
b = data[pos++];
if (b == 0)
productId = 0;
else if(b != 1)
throw new InvalidDataException("Invalid ProductID");
else
productId = data[pos];
pos += b;
break;
case (byte) SerialNumberChunks.MaxBuild:
maxBuildDt = new DateTime(data[pos + 2] + 256 * data[pos + 3], data[pos + 1],data[pos]);
pos += 4;
break;
case (byte) SerialNumberChunks.End:
if (pos + 4 > data.Length)
throw new InvalidDataException("No checksum");
{
SHA1 sha = new SHA1Managed();
sha.Initialize();
var hash = sha.ComputeHash(data, i, pos - 1 - i);
for (int j = 0; j < 4; j++)
{
if(data[pos + j] == hash[3 - j])
continue;
throw new InvalidDataException("Invalid checksum");
}
}
return;
}
}
throw new InvalidDataException("No checksum");
}
public static string GenerateKey(int productId, string customerName, string eMail, DateTime maxBuildDt)
{
var data = new MemoryStream();
data.WriteByte((byte)SerialNumberChunks.Version);
data.WriteByte(1);
data.WriteByte((byte)SerialNumberChunks.UserName);
var utfCustomer = Encoding.UTF8.GetBytes(customerName);
if (utfCustomer.Length > 255)
throw new ArgumentException("Customer name too long", "customerName");
data.WriteByte((byte)utfCustomer.Length);
data.Write(utfCustomer, 0, utfCustomer.Length);
data.WriteByte((byte)SerialNumberChunks.Email);
byte[] utfeMail = Encoding.UTF8.GetBytes(eMail);
if (utfeMail.Length > 255)
throw new ArgumentException("EMail too long", "eMail");
data.WriteByte((byte)utfeMail.Length);
data.Write(utfeMail, 0, utfeMail.Length);
data.WriteByte((byte)SerialNumberChunks.ProductCode);
data.Write(new byte[] { 41, 65, 36, 150, 5, 175, 174, 137 }, 0, 8);
data.WriteByte((byte)SerialNumberChunks.UserData);
data.WriteByte(1);
data.WriteByte((byte)productId);
data.WriteByte((byte)SerialNumberChunks.MaxBuild);
data.WriteByte((byte)maxBuildDt.Day);
data.WriteByte((byte)maxBuildDt.Month);
data.WriteByte((byte)maxBuildDt.Year);
data.WriteByte((byte)(maxBuildDt.Year >> 8));
SHA1 sha = new SHA1Managed();
sha.Initialize();
data.Position = 0;
var hash = sha.ComputeHash(data);
data.WriteByte((byte)SerialNumberChunks.End);
data.WriteByte(hash[3]);
data.WriteByte(hash[2]);
data.WriteByte(hash[1]);
data.WriteByte(hash[0]);
const int minPadding = 8 + 3;
const int maxPadding = minPadding + 16;
const int maxBytes = 3072 / 8;
if (data.Length + minPadding > maxBytes)
throw new ApplicationException("Serial number too long");
var rnd = new Random();
var paddingBytes = rnd.Next(minPadding, maxPadding + 1);
if (data.Length + paddingBytes > maxBytes)
paddingBytes = maxBytes - (int)data.Length;
var paddedData = new byte[maxBytes];
var nonPaddedData = data.ToArray();
Array.Copy(nonPaddedData, paddedData, paddingBytes);
Array.Copy(nonPaddedData, 0, paddedData, paddingBytes, data.Length);
paddedData[0] = 0;
paddedData[1] = 2;
paddedData[paddingBytes - 1] = 0;
var i = 2;
for (; i < paddingBytes - 1; i++) {
byte b = 0;
while (b == 0) {
b = (byte)rnd.Next(256);
}
paddedData[i] = b;
}
i = nonPaddedData.Length + paddingBytes;
while (i < maxBytes) {
paddedData[i++] = (byte)rnd.Next(256);
}
var res = Convert.ToBase64String(Rsa.Encrypt(paddedData), Base64FormattingOptions.InsertLineBreaks);
return res;
}
}
}