65 lines
2.2 KiB
C#
65 lines
2.2 KiB
C#
|
using NUnit.Framework;
|
|||
|
using System;
|
|||
|
using System.Collections.Generic;
|
|||
|
|
|||
|
namespace Obfuz.EncryptionVM.Instructions
|
|||
|
{
|
|||
|
public class MultipleInstruction : EncryptionInstructionBase
|
|||
|
{
|
|||
|
private readonly int _multiValue;
|
|||
|
private readonly int _revertMultiValue;
|
|||
|
private readonly int _opKeyIndex;
|
|||
|
|
|||
|
public MultipleInstruction(int addValue, int opKeyIndex)
|
|||
|
{
|
|||
|
_multiValue = addValue;
|
|||
|
_opKeyIndex = opKeyIndex;
|
|||
|
_revertMultiValue = (int)ModInverseOdd((uint)addValue);
|
|||
|
Verify();
|
|||
|
}
|
|||
|
|
|||
|
private void Verify()
|
|||
|
{
|
|||
|
int a = 1122334;
|
|||
|
Assert.AreEqual(a, a * _multiValue * _revertMultiValue);
|
|||
|
}
|
|||
|
|
|||
|
public static uint ModInverseOdd(uint a)
|
|||
|
{
|
|||
|
if (a % 2 == 0)
|
|||
|
throw new ArgumentException("Input must be an odd number.", nameof(a));
|
|||
|
|
|||
|
uint x = 1; // 初始解:x₀ = 1 (mod 2)
|
|||
|
for (int i = 0; i < 5; i++) // 迭代5次(2^1 → 2^32)
|
|||
|
{
|
|||
|
int shift = 2 << i; // 当前模数为 2^(2^(i+1))
|
|||
|
ulong mod = 1UL << shift; // 使用 ulong 避免溢出
|
|||
|
ulong ax = (ulong)a * x; // 计算 a*x(64位避免截断)
|
|||
|
ulong term = (2 - ax) % mod;
|
|||
|
x = (uint)((x * term) % mod); // 更新 x,结果截断为 uint
|
|||
|
}
|
|||
|
return x; // 最终解为 x₅ mod 2^32
|
|||
|
}
|
|||
|
|
|||
|
public override int Encrypt(int value, int[] secretKey, int salt)
|
|||
|
{
|
|||
|
return value * _multiValue + secretKey[_opKeyIndex] + salt;
|
|||
|
}
|
|||
|
|
|||
|
public override int Decrypt(int value, int[] secretKey, int salt)
|
|||
|
{
|
|||
|
return (value - secretKey[_opKeyIndex] - salt) * _revertMultiValue;
|
|||
|
}
|
|||
|
|
|||
|
public override void GenerateEncryptCode(List<string> lines, string indent)
|
|||
|
{
|
|||
|
lines.Add(indent + $"value = value * {_multiValue} + _secretKey[{_opKeyIndex}] + salt;");
|
|||
|
}
|
|||
|
|
|||
|
public override void GenerateDecryptCode(List<string> lines, string indent)
|
|||
|
{
|
|||
|
lines.Add(indent + $"value = (value - _secretKey[{_opKeyIndex}] - salt) * {_revertMultiValue};");
|
|||
|
}
|
|||
|
}
|
|||
|
}
|