using System; using System.IO; using libhac.XTSSharp; namespace libhac { public class Aes128CtrStream : SectorStream { private const int CryptChunkSize = 0x4000; private readonly long _counterOffset; private readonly byte[] _tempBuffer; private readonly Aes128CtrTransform _decryptor; protected readonly byte[] Counter; /// /// Creates a new stream /// /// The base stream /// The decryption key /// Offset to add to the counter /// The value of the upper 64 bits of the counter public Aes128CtrStream(Stream baseStream, byte[] key, long counterOffset = 0, byte[] ctrHi = null) : this(baseStream, key, 0, baseStream.Length, counterOffset, ctrHi) { } /// /// Creates a new stream /// /// The base stream /// The decryption key /// The intial counter public Aes128CtrStream(Stream baseStream, byte[] key, byte[] counter) : base(baseStream, 0x10, 0) { _counterOffset = 0; Length = baseStream.Length; _tempBuffer = new byte[CryptChunkSize]; _decryptor = new Aes128CtrTransform(key, counter ?? new byte[0x10], CryptChunkSize); Counter = _decryptor.Counter; } /// /// Creates a new stream /// /// The base stream /// The decryption key /// Offset to start at in the input stream /// The length of the created stream /// Offset to add to the counter /// The value of the upper 64 bits of the counter public Aes128CtrStream(Stream baseStream, byte[] key, long offset, long length, long counterOffset, byte[] ctrHi = null) : base(baseStream, CryptChunkSize, offset) { var initialCounter = new byte[0x10]; if (ctrHi != null) { Array.Copy(ctrHi, initialCounter, 8); } _counterOffset = counterOffset; Length = length; _tempBuffer = new byte[CryptChunkSize]; _decryptor = new Aes128CtrTransform(key, initialCounter, CryptChunkSize); Counter = _decryptor.Counter; UpdateCounter(_counterOffset + base.Position); } private void UpdateCounter(long offset) { offset >>= 4; for (uint j = 0; j < 0x8; j++) { Counter[0x10 - j - 1] = (byte)(offset & 0xFF); offset >>= 8; } } public override void Flush() { throw new NotImplementedException(); } public override long Seek(long offset, SeekOrigin origin) { switch (origin) { case SeekOrigin.Begin: Position = offset; break; case SeekOrigin.Current: Position += offset; break; case SeekOrigin.End: Position = Length - offset; break; } return Position; } public override void SetLength(long value) { throw new NotSupportedException(); } public override void Write(byte[] buffer, int offset, int count) { throw new NotImplementedException(); } public override bool CanRead => true; public override bool CanSeek => true; public override bool CanWrite => false; public override long Length { get; } public override long Position { get => base.Position; set { base.Position = value; UpdateCounter(_counterOffset + base.Position); } } public override int Read(byte[] buffer, int offset, int count) { ValidateSize(count); var bytesRead = base.Read(_tempBuffer, 0, count); if (bytesRead == 0) return 0; return _decryptor.TransformBlock(_tempBuffer, 0, bytesRead, buffer, offset); } protected override void ValidateSizeMultiple(long value) { if (value % 0x10 != 0) throw new ArgumentException($"Value needs to be a multiple of {SectorSize}"); } } }