2019-08-28 00:39:15 +01:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.Linq;
|
|
|
|
|
|
|
|
|
|
namespace libsharperang {
|
|
|
|
|
public class CRC32 {
|
2019-08-28 01:06:10 +01:00
|
|
|
|
private uint _Initial = 0xFFFFFFFF;
|
2019-09-10 00:43:58 +01:00
|
|
|
|
public uint Initial { get => ~_Initial; private set => _Initial=~value; }
|
2019-08-28 00:39:15 +01:00
|
|
|
|
private uint Polynomial = 0xedb88320;
|
|
|
|
|
private uint[] Table;
|
|
|
|
|
public bool Initialised=false;
|
|
|
|
|
public CRC32(uint Initial) => this.Initial=Initial;
|
|
|
|
|
public CRC32() { }
|
|
|
|
|
public void Initialise() {
|
|
|
|
|
Table=Enumerable.Range(0, 256).Select(i => {
|
|
|
|
|
uint e=(uint)i;
|
|
|
|
|
for (ushort eb = 0; eb<8; ++eb) {
|
|
|
|
|
e=((e&1)!=0)
|
|
|
|
|
? (Polynomial^(e>>1))
|
|
|
|
|
: (e>>1);
|
|
|
|
|
}
|
|
|
|
|
return e;
|
|
|
|
|
}).ToArray();
|
|
|
|
|
Initialised=true;
|
|
|
|
|
}
|
|
|
|
|
public uint Reflect(uint iv) {
|
|
|
|
|
uint ivr=0;
|
|
|
|
|
for (int i = 0; i<32; i++) {
|
|
|
|
|
uint bit=(iv>>i)&1;
|
|
|
|
|
ivr |= (bit<<(31-i));
|
|
|
|
|
}
|
|
|
|
|
return ~ivr;
|
|
|
|
|
}
|
|
|
|
|
public uint GetChecksum<T>(IEnumerable<T> bytes) {
|
|
|
|
|
if (!Initialised) Initialise();
|
|
|
|
|
try {
|
2019-08-28 01:06:10 +01:00
|
|
|
|
return ~bytes.Aggregate(_Initial,
|
2019-08-28 00:39:15 +01:00
|
|
|
|
(csR, cB) => Table[(csR & 0xFF) ^ Convert.ToByte(cB)] ^ (csR>>8));
|
|
|
|
|
} catch (FormatException e) {
|
|
|
|
|
throw new Exception("Could not read stream as bytes", e);
|
|
|
|
|
} catch (InvalidCastException e) {
|
|
|
|
|
throw new Exception("Could not read stream as bytes", e);
|
|
|
|
|
} catch (OverflowException e) {
|
|
|
|
|
throw new Exception("Could not read stream as bytes", e);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|