aboutsummaryrefslogtreecommitdiff
path: root/utils/ipn_sqlclr/keygen.cs
diff options
context:
space:
mode:
authorjmpoep <OriginalEntryPoint@qq.com>2023-12-07 16:51:07 +0800
committerjmpoep <OriginalEntryPoint@qq.com>2023-12-07 16:51:07 +0800
commit28008a746a31abb7909dd86cb0cd413ac8943b0b (patch)
treea30b74b8cad548048c3c1551d652828ab76fa9bd /utils/ipn_sqlclr/keygen.cs
downloadvmprotect-3.5.1-28008a746a31abb7909dd86cb0cd413ac8943b0b.tar
vmprotect-3.5.1-28008a746a31abb7909dd86cb0cd413ac8943b0b.tar.gz
vmprotect-3.5.1-28008a746a31abb7909dd86cb0cd413ac8943b0b.tar.bz2
vmprotect-3.5.1-28008a746a31abb7909dd86cb0cd413ac8943b0b.zip
first commitHEADmaster
Diffstat (limited to 'utils/ipn_sqlclr/keygen.cs')
-rw-r--r--utils/ipn_sqlclr/keygen.cs212
1 files changed, 212 insertions, 0 deletions
diff --git a/utils/ipn_sqlclr/keygen.cs b/utils/ipn_sqlclr/keygen.cs
new file mode 100644
index 0000000..d26b5a3
--- /dev/null
+++ b/utils/ipn_sqlclr/keygen.cs
@@ -0,0 +1,212 @@
+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;
+ }
+ }
+}