mirror of
https://github.com/MikaylaFischler/cc-mek-scada.git
synced 2024-08-30 18:22:34 +00:00
174 lines
4.5 KiB
Lua
174 lines
4.5 KiB
Lua
|
require("lockbox").insecure();
|
||
|
|
||
|
local Bit = require("lockbox.util.bit");
|
||
|
local String = require("string");
|
||
|
local Math = require("math");
|
||
|
local Queue = require("lockbox.util.queue");
|
||
|
|
||
|
local AND = Bit.band;
|
||
|
local OR = Bit.bor;
|
||
|
local XOR = Bit.bxor;
|
||
|
local LROT = Bit.lrotate;
|
||
|
local LSHIFT = Bit.lshift;
|
||
|
local RSHIFT = Bit.rshift;
|
||
|
|
||
|
--SHA1 is big-endian
|
||
|
local bytes2word = function(b0, b1, b2, b3)
|
||
|
local i = b0; i = LSHIFT(i, 8);
|
||
|
i = OR(i, b1); i = LSHIFT(i, 8);
|
||
|
i = OR(i, b2); i = LSHIFT(i, 8);
|
||
|
i = OR(i, b3);
|
||
|
return i;
|
||
|
end
|
||
|
|
||
|
local word2bytes = function(word)
|
||
|
local b0, b1, b2, b3;
|
||
|
b3 = AND(word, 0xFF); word = RSHIFT(word, 8);
|
||
|
b2 = AND(word, 0xFF); word = RSHIFT(word, 8);
|
||
|
b1 = AND(word, 0xFF); word = RSHIFT(word, 8);
|
||
|
b0 = AND(word, 0xFF);
|
||
|
return b0, b1, b2, b3;
|
||
|
end
|
||
|
|
||
|
local dword2bytes = function(i)
|
||
|
local b4, b5, b6, b7 = word2bytes(i);
|
||
|
local b0, b1, b2, b3 = word2bytes(Math.floor(i / 0x100000000));
|
||
|
return b0, b1, b2, b3, b4, b5, b6, b7;
|
||
|
end
|
||
|
|
||
|
local F = function(x, y, z) return XOR(z, AND(x, XOR(y, z))); end
|
||
|
local G = function(x, y, z) return XOR(x, XOR(y, z)); end
|
||
|
local H = function(x, y, z) return OR(AND(x, OR(y, z)), AND(y, z)); end
|
||
|
local I = function(x, y, z) return XOR(x, XOR(y, z)); end
|
||
|
|
||
|
local SHA1 = function()
|
||
|
|
||
|
local queue = Queue();
|
||
|
|
||
|
local h0 = 0x67452301;
|
||
|
local h1 = 0xEFCDAB89;
|
||
|
local h2 = 0x98BADCFE;
|
||
|
local h3 = 0x10325476;
|
||
|
local h4 = 0xC3D2E1F0;
|
||
|
|
||
|
local public = {};
|
||
|
|
||
|
local processBlock = function()
|
||
|
local a = h0;
|
||
|
local b = h1;
|
||
|
local c = h2;
|
||
|
local d = h3;
|
||
|
local e = h4;
|
||
|
local temp;
|
||
|
local k;
|
||
|
|
||
|
local w = {};
|
||
|
for i = 0, 15 do
|
||
|
w[i] = bytes2word(queue.pop(), queue.pop(), queue.pop(), queue.pop());
|
||
|
end
|
||
|
|
||
|
for i = 16, 79 do
|
||
|
w[i] = LROT((XOR(XOR(w[i - 3], w[i - 8]), XOR(w[i - 14], w[i - 16]))), 1);
|
||
|
end
|
||
|
|
||
|
for i = 0, 79 do
|
||
|
if (i <= 19) then
|
||
|
temp = F(b, c, d);
|
||
|
k = 0x5A827999;
|
||
|
elseif (i <= 39) then
|
||
|
temp = G(b, c, d);
|
||
|
k = 0x6ED9EBA1;
|
||
|
elseif (i <= 59) then
|
||
|
temp = H(b, c, d);
|
||
|
k = 0x8F1BBCDC;
|
||
|
else
|
||
|
temp = I(b, c, d);
|
||
|
k = 0xCA62C1D6;
|
||
|
end
|
||
|
temp = LROT(a, 5) + temp + e + k + w[i];
|
||
|
e = d;
|
||
|
d = c;
|
||
|
c = LROT(b, 30);
|
||
|
b = a;
|
||
|
a = temp;
|
||
|
end
|
||
|
|
||
|
h0 = AND(h0 + a, 0xFFFFFFFF);
|
||
|
h1 = AND(h1 + b, 0xFFFFFFFF);
|
||
|
h2 = AND(h2 + c, 0xFFFFFFFF);
|
||
|
h3 = AND(h3 + d, 0xFFFFFFFF);
|
||
|
h4 = AND(h4 + e, 0xFFFFFFFF);
|
||
|
end
|
||
|
|
||
|
public.init = function()
|
||
|
queue.reset();
|
||
|
h0 = 0x67452301;
|
||
|
h1 = 0xEFCDAB89;
|
||
|
h2 = 0x98BADCFE;
|
||
|
h3 = 0x10325476;
|
||
|
h4 = 0xC3D2E1F0;
|
||
|
return public;
|
||
|
end
|
||
|
|
||
|
|
||
|
public.update = function(bytes)
|
||
|
for b in bytes do
|
||
|
queue.push(b);
|
||
|
if queue.size() >= 64 then processBlock(); end
|
||
|
end
|
||
|
|
||
|
return public;
|
||
|
end
|
||
|
|
||
|
public.finish = function()
|
||
|
local bits = queue.getHead() * 8;
|
||
|
|
||
|
queue.push(0x80);
|
||
|
while ((queue.size() + 7) % 64) < 63 do
|
||
|
queue.push(0x00);
|
||
|
end
|
||
|
|
||
|
local b0, b1, b2, b3, b4, b5, b6, b7 = dword2bytes(bits);
|
||
|
|
||
|
queue.push(b0);
|
||
|
queue.push(b1);
|
||
|
queue.push(b2);
|
||
|
queue.push(b3);
|
||
|
queue.push(b4);
|
||
|
queue.push(b5);
|
||
|
queue.push(b6);
|
||
|
queue.push(b7);
|
||
|
|
||
|
while queue.size() > 0 do
|
||
|
processBlock();
|
||
|
end
|
||
|
|
||
|
return public;
|
||
|
end
|
||
|
|
||
|
public.asBytes = function()
|
||
|
local b0, b1, b2, b3 = word2bytes(h0);
|
||
|
local b4, b5, b6, b7 = word2bytes(h1);
|
||
|
local b8, b9, b10, b11 = word2bytes(h2);
|
||
|
local b12, b13, b14, b15 = word2bytes(h3);
|
||
|
local b16, b17, b18, b19 = word2bytes(h4);
|
||
|
|
||
|
return {b0, b1, b2, b3, b4, b5, b6, b7, b8, b9, b10, b11, b12, b13, b14, b15, b16, b17, b18, b19};
|
||
|
end
|
||
|
|
||
|
public.asHex = function()
|
||
|
local b0, b1, b2, b3 = word2bytes(h0);
|
||
|
local b4, b5, b6, b7 = word2bytes(h1);
|
||
|
local b8, b9, b10, b11 = word2bytes(h2);
|
||
|
local b12, b13, b14, b15 = word2bytes(h3);
|
||
|
local b16, b17, b18, b19 = word2bytes(h4);
|
||
|
|
||
|
return String.format("%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x%02x",
|
||
|
b0, b1, b2, b3, b4, b5, b6, b7, b8, b9, b10, b11, b12, b13, b14, b15, b16, b17, b18, b19);
|
||
|
end
|
||
|
|
||
|
return public;
|
||
|
end
|
||
|
|
||
|
return SHA1;
|