188 lines
5.1 KiB
C#
188 lines
5.1 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using SharpCompress.Compressors.LZMA;
|
|
|
|
namespace SharpCompress.Common.SevenZip
|
|
{
|
|
internal class CFolder
|
|
{
|
|
internal List<CCoderInfo> _coders = new List<CCoderInfo>();
|
|
internal List<CBindPair> _bindPairs = new List<CBindPair>();
|
|
internal List<int> _packStreams = new List<int>();
|
|
internal int _firstPackStreamId;
|
|
internal List<long> _unpackSizes = new List<long>();
|
|
internal uint? _unpackCrc;
|
|
|
|
internal bool UnpackCrcDefined => _unpackCrc != null;
|
|
|
|
public long GetUnpackSize()
|
|
{
|
|
if (_unpackSizes.Count == 0)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
for (int i = _unpackSizes.Count - 1; i >= 0; i--)
|
|
{
|
|
if (FindBindPairForOutStream(i) < 0)
|
|
{
|
|
return _unpackSizes[i];
|
|
}
|
|
}
|
|
|
|
throw new Exception();
|
|
}
|
|
|
|
public int GetNumOutStreams()
|
|
{
|
|
int count = 0;
|
|
for (int i = 0; i < _coders.Count; i++)
|
|
{
|
|
count += _coders[i]._numOutStreams;
|
|
}
|
|
|
|
return count;
|
|
}
|
|
|
|
public int FindBindPairForInStream(int inStreamIndex)
|
|
{
|
|
for (int i = 0; i < _bindPairs.Count; i++)
|
|
{
|
|
if (_bindPairs[i]._inIndex == inStreamIndex)
|
|
{
|
|
return i;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
public int FindBindPairForOutStream(int outStreamIndex)
|
|
{
|
|
for (int i = 0; i < _bindPairs.Count; i++)
|
|
{
|
|
if (_bindPairs[i]._outIndex == outStreamIndex)
|
|
{
|
|
return i;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
public int FindPackStreamArrayIndex(int inStreamIndex)
|
|
{
|
|
for (int i = 0; i < _packStreams.Count; i++)
|
|
{
|
|
if (_packStreams[i] == inStreamIndex)
|
|
{
|
|
return i;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
public bool IsEncrypted()
|
|
{
|
|
for (int i = _coders.Count - 1; i >= 0; i--)
|
|
{
|
|
if (_coders[i]._methodId == CMethodId.K_AES)
|
|
{
|
|
return true;
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
public bool CheckStructure()
|
|
{
|
|
const int kNumCodersMax = 32; // don't change it
|
|
const int kMaskSize = 32; // it must be >= kNumCodersMax
|
|
const int kNumBindsMax = 32;
|
|
|
|
if (_coders.Count > kNumCodersMax || _bindPairs.Count > kNumBindsMax)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
{
|
|
var v = new BitVector(_bindPairs.Count + _packStreams.Count);
|
|
|
|
for (int i = 0; i < _bindPairs.Count; i++)
|
|
{
|
|
if (v.GetAndSet(_bindPairs[i]._inIndex))
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < _packStreams.Count; i++)
|
|
{
|
|
if (v.GetAndSet(_packStreams[i]))
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
{
|
|
var v = new BitVector(_unpackSizes.Count);
|
|
for (int i = 0; i < _bindPairs.Count; i++)
|
|
{
|
|
if (v.GetAndSet(_bindPairs[i]._outIndex))
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
uint[] mask = new uint[kMaskSize];
|
|
|
|
{
|
|
List<int> inStreamToCoder = new List<int>();
|
|
List<int> outStreamToCoder = new List<int>();
|
|
for (int i = 0; i < _coders.Count; i++)
|
|
{
|
|
CCoderInfo coder = _coders[i];
|
|
for (int j = 0; j < coder._numInStreams; j++)
|
|
{
|
|
inStreamToCoder.Add(i);
|
|
}
|
|
for (int j = 0; j < coder._numOutStreams; j++)
|
|
{
|
|
outStreamToCoder.Add(i);
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < _bindPairs.Count; i++)
|
|
{
|
|
CBindPair bp = _bindPairs[i];
|
|
mask[inStreamToCoder[bp._inIndex]] |= (1u << outStreamToCoder[bp._outIndex]);
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < kMaskSize; i++)
|
|
{
|
|
for (int j = 0; j < kMaskSize; j++)
|
|
{
|
|
if (((1u << j) & mask[i]) != 0)
|
|
{
|
|
mask[i] |= mask[j];
|
|
}
|
|
}
|
|
}
|
|
|
|
for (int i = 0; i < kMaskSize; i++)
|
|
{
|
|
if (((1u << i) & mask[i]) != 0)
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
}
|
|
} |