123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170 |
- #if !NET_1_1 && !NETCF_2_0
- using System;
- using System.IO;
- using System.Security.Cryptography;
- namespace CommonMPQ.SharpZipLib.Encryption {
-
-
-
-
-
- internal class ZipAESStream : CryptoStream {
-
-
-
-
-
-
- public ZipAESStream(Stream stream, ZipAESTransform transform, CryptoStreamMode mode)
- : base(stream, transform, mode) {
- _stream = stream;
- _transform = transform;
- _slideBuffer = new byte[1024];
- _blockAndAuth = CRYPTO_BLOCK_SIZE + AUTH_CODE_LENGTH;
-
-
-
- if (mode != CryptoStreamMode.Read) {
- throw new Exception("ZipAESStream only for read");
- }
- }
-
- private const int AUTH_CODE_LENGTH = 10;
- private Stream _stream;
- private ZipAESTransform _transform;
- private byte[] _slideBuffer;
- private int _slideBufStartPos;
- private int _slideBufFreePos;
-
- private const int CRYPTO_BLOCK_SIZE = 16;
- private int _blockAndAuth;
-
-
-
-
- public override int Read(byte[] outBuffer, int offset, int count) {
- int nBytes = 0;
- while (nBytes < count) {
-
- int byteCount = _slideBufFreePos - _slideBufStartPos;
-
-
-
- int lengthToRead = _blockAndAuth - byteCount;
- if (_slideBuffer.Length - _slideBufFreePos < lengthToRead) {
-
- int iTo = 0;
- for (int iFrom = _slideBufStartPos; iFrom < _slideBufFreePos; iFrom++, iTo++) {
- _slideBuffer[iTo] = _slideBuffer[iFrom];
- }
- _slideBufFreePos -= _slideBufStartPos;
- _slideBufStartPos = 0;
- }
- int obtained = _stream.Read(_slideBuffer, _slideBufFreePos, lengthToRead);
- _slideBufFreePos += obtained;
-
- byteCount = _slideBufFreePos - _slideBufStartPos;
- if (byteCount >= _blockAndAuth) {
-
- _transform.TransformBlock(_slideBuffer,
- _slideBufStartPos,
- CRYPTO_BLOCK_SIZE,
- outBuffer,
- offset);
- nBytes += CRYPTO_BLOCK_SIZE;
- offset += CRYPTO_BLOCK_SIZE;
- _slideBufStartPos += CRYPTO_BLOCK_SIZE;
- } else {
-
- if (byteCount > AUTH_CODE_LENGTH) {
-
- int finalBlock = byteCount - AUTH_CODE_LENGTH;
- _transform.TransformBlock(_slideBuffer,
- _slideBufStartPos,
- finalBlock,
- outBuffer,
- offset);
- nBytes += finalBlock;
- _slideBufStartPos += finalBlock;
- }
- else if (byteCount < AUTH_CODE_LENGTH)
- throw new Exception("Internal error missed auth code");
-
- byte[] calcAuthCode = _transform.GetAuthCode();
- for (int i = 0; i < AUTH_CODE_LENGTH; i++) {
- if (calcAuthCode[i] != _slideBuffer[_slideBufStartPos + i]) {
- throw new Exception("AES Authentication Code does not match. This is a super-CRC check on the data in the file after compression and encryption. \r\n"
- + "The file may be damaged.");
- }
- }
- break;
- }
- }
- return nBytes;
- }
-
-
-
-
-
-
- public override void Write(byte[] buffer, int offset, int count) {
-
- throw new NotImplementedException();
- }
- }
- }
- #endif
|