?? udfin.cpp
字號:
if (extendedAttrLen > size - pos)
return S_FALSE;
/*
if (extendedAttrLen != 16)
{
if (extendedAttrLen < 24)
return S_FALSE;
CTag attrTag;
RINOK(attrTag.Parse(p + pos, size));
if (attrTag.Id != DESC_TYPE_ExtendedAttrHeader)
return S_FALSE;
// UInt32 implAttrLocation = Get32(p + pos + 16);
// UInt32 applicationlAttrLocation = Get32(p + pos + 20);
}
*/
pos += extendedAttrLen;
int desctType = item.IcbTag.GetDescriptorType();
if (allocDescriptorsLen > size - pos)
return S_FALSE;
if (desctType == ICB_DESC_TYPE_INLINE)
{
item.IsInline = true;
item.InlineData.SetCapacity(allocDescriptorsLen);
memcpy(item.InlineData, p + pos, allocDescriptorsLen);
}
else
{
item.IsInline = false;
if (desctType != ICB_DESC_TYPE_SHORT && desctType != ICB_DESC_TYPE_LONG)
return S_FALSE;
for (UInt32 i = 0; i < allocDescriptorsLen;)
{
CMyExtent e;
if (desctType == ICB_DESC_TYPE_SHORT)
{
if (i + 8 > allocDescriptorsLen)
return S_FALSE;
CShortAllocDesc sad;
sad.Parse(p + pos + i);
e.Pos = sad.Pos;
e.Len = sad.Len;
e.PartitionRef = lad.Location.PartitionRef;
i += 8;
}
else
{
if (i + 16 > allocDescriptorsLen)
return S_FALSE;
CLongAllocDesc ladNew;
ladNew.Parse(p + pos + i);
e.Pos = ladNew.Location.Pos;
e.PartitionRef = ladNew.Location.PartitionRef;
e.Len = ladNew.Len;
i += 16;
}
item.Extents.Add(e);
}
}
if (item.IcbTag.IsDir())
{
if (!item.CheckChunkSizes() || !CheckItemExtents(volIndex, item))
return S_FALSE;
CByteBuffer buf;
RINOK(ReadFromFile(volIndex, item, buf));
item.Size = 0;
item.Extents.ClearAndFree();
item.InlineData.Free();
const Byte *p = buf;
size = buf.GetCapacity();
size_t processedTotal = 0;
for (; processedTotal < size;)
{
size_t processedCur;
CFileId fileId;
RINOK(fileId.Parse(p + processedTotal, size - processedTotal, processedCur));
if (!fileId.IsItLinkParent())
{
CFile file;
// file.FileVersion = fileId.FileVersion;
// file.FileCharacteristics = fileId.FileCharacteristics;
// file.ImplUse = fileId.ImplUse;
file.Id = fileId.Id;
_fileNameLengthTotal += file.Id.Data.GetCapacity();
if (_fileNameLengthTotal > kFileNameLengthTotalMax)
return S_FALSE;
item.SubFiles.Add(Files.Size());
if (Files.Size() > kNumFilesMax)
return S_FALSE;
Files.Add(file);
RINOK(ReadFileItem(volIndex, fsIndex, fileId.Icb, numRecurseAllowed));
}
processedTotal += processedCur;
}
}
else
{
if ((UInt32)item.Extents.Size() > kNumExtentsMax - _numExtents)
return S_FALSE;
_numExtents += item.Extents.Size();
if (item.InlineData.GetCapacity() > kInlineExtentsSizeMax - _inlineExtentsSize)
return S_FALSE;
_inlineExtentsSize += item.InlineData.GetCapacity();
}
return S_OK;
}
HRESULT CInArchive::FillRefs(CFileSet &fs, int fileIndex, int parent, int numRecurseAllowed)
{
if (_numRefs % 10000 == 0)
{
RINOK(_progress->SetCompleted());
}
if (numRecurseAllowed-- == 0)
return S_FALSE;
if (_numRefs >= kNumRefsMax)
return S_FALSE;
_numRefs++;
CRef ref;
ref.FileIndex = fileIndex;
ref.Parent = parent;
parent = fs.Refs.Size();
fs.Refs.Add(ref);
const CItem &item = Items[Files[fileIndex].ItemIndex];
for (int i = 0; i < item.SubFiles.Size(); i++)
{
RINOK(FillRefs(fs, item.SubFiles[i], parent, numRecurseAllowed));
}
return S_OK;
}
HRESULT CInArchive::Open2()
{
Clear();
UInt64 fileSize;
RINOK(_stream->Seek(0, STREAM_SEEK_END, &fileSize));
const int kSecLogSizeMax = 11;
const int kSecLogSizeMin = 8;
Byte buf[1 << kSecLogSizeMax];
for (SecLogSize = kSecLogSizeMax; SecLogSize >= kSecLogSizeMin; SecLogSize -= 3)
{
Int32 bufSize = 1 << SecLogSize;
if (bufSize > fileSize)
return S_FALSE;
RINOK(_stream->Seek(-bufSize, STREAM_SEEK_END, NULL));
RINOK(ReadStream_FALSE(_stream, buf, bufSize));
CTag tag;
if (tag.Parse(buf, bufSize) == S_OK)
if (tag.Id == DESC_TYPE_AnchorVolPtr)
break;
}
if (SecLogSize < kSecLogSizeMin)
return S_FALSE;
CExtent extentVDS;
extentVDS.Parse(buf + 16);
for (UInt32 location = extentVDS.Pos; ; location++)
{
size_t bufSize = 1 << SecLogSize;
size_t pos = 0;
RINOK(_stream->Seek((UInt64)location << SecLogSize, STREAM_SEEK_SET, NULL));
RINOK(ReadStream_FALSE(_stream, buf, bufSize));
CTag tag;
RINOK(tag.Parse(buf + pos, bufSize - pos));
if (tag.Id == DESC_TYPE_Terminating)
break;
if (tag.Id == DESC_TYPE_Partition)
{
if (Partitions.Size() >= kNumPartitionsMax)
return S_FALSE;
CPartition partition;
// UInt32 volDescSeqNumer = Get32(buf + 16);
// partition.Flags = Get16(buf + 20);
partition.Number = Get16(buf + 22);
// partition.ContentsId.Parse(buf + 24);
// memcpy(partition.ContentsUse, buf + 56, sizeof(partition.ContentsUse));
// ContentsUse is Partition Header Description.
// partition.AccessType = Get32(buf + 184);
partition.Pos = Get32(buf + 188);
partition.Len = Get32(buf + 192);
// partition.ImplId.Parse(buf + 196);
// memcpy(partition.ImplUse, buf + 228, sizeof(partition.ImplUse));
Partitions.Add(partition);
}
else if (tag.Id == DESC_TYPE_LogicalVol)
{
if (LogVols.Size() >= kNumLogVolumesMax)
return S_FALSE;
CLogVol vol;
vol.Id.Parse(buf + 84);
vol.BlockSize = Get32(buf + 212);
// vol.DomainId.Parse(buf + 216);
if (vol.BlockSize < 512 || vol.BlockSize > ((UInt32)1 << 30))
return S_FALSE;
// memcpy(vol.ContentsUse, buf + 248, sizeof(vol.ContentsUse));
vol.FileSetLocation.Parse(buf + 248);
// UInt32 mapTableLength = Get32(buf + 264);
UInt32 numPartitionMaps = Get32(buf + 268);
if (numPartitionMaps > kNumPartitionsMax)
return S_FALSE;
// vol.ImplId.Parse(buf + 272);
// memcpy(vol.ImplUse, buf + 128, sizeof(vol.ImplUse));
size_t pos = 440;
for (UInt32 i = 0; i < numPartitionMaps; i++)
{
if (pos + 2 > bufSize)
return S_FALSE;
CPartitionMap pm;
pm.Type = buf[pos];
// pm.Length = buf[pos + 1];
Byte len = buf[pos + 1];
if (pos + len > bufSize)
return S_FALSE;
// memcpy(pm.Data, buf + pos + 2, pm.Length - 2);
if (pm.Type == 1)
{
if (pos + 6 > bufSize)
return S_FALSE;
// pm.VolSeqNumber = Get16(buf + pos + 2);
pm.PartitionNumber = Get16(buf + pos + 4);
}
else
return S_FALSE;
pos += len;
vol.PartitionMaps.Add(pm);
}
LogVols.Add(vol);
}
}
UInt64 totalSize = 0;
int volIndex;
for (volIndex = 0; volIndex < LogVols.Size(); volIndex++)
{
CLogVol &vol = LogVols[volIndex];
for (int pmIndex = 0; pmIndex < vol.PartitionMaps.Size(); pmIndex++)
{
CPartitionMap &pm = vol.PartitionMaps[pmIndex];
int i;
for (i = 0; i < Partitions.Size(); i++)
{
CPartition &part = Partitions[i];
if (part.Number == pm.PartitionNumber)
{
if (part.VolIndex >= 0)
return S_FALSE;
pm.PartitionIndex = i;
part.VolIndex = volIndex;
totalSize += (UInt64)part.Len << SecLogSize;
break;
}
}
if (i == Partitions.Size())
return S_FALSE;
}
}
RINOK(_progress->SetTotal(totalSize));
for (volIndex = 0; volIndex < LogVols.Size(); volIndex++)
{
CLogVol &vol = LogVols[volIndex];
CLongAllocDesc nextExtent = vol.FileSetLocation;
// while (nextExtent.ExtentLen != 0)
// for (int i = 0; i < 1; i++)
{
if (nextExtent.GetLen() < 512)
return S_FALSE;
CByteBuffer buf;
buf.SetCapacity(nextExtent.GetLen());
RINOK(Read(volIndex, nextExtent, buf));
const Byte *p = buf;
size_t size = nextExtent.GetLen();
CTag tag;
RINOK(tag.Parse(p, size));
if (tag.Id != DESC_TYPE_FileSet)
return S_FALSE;
CFileSet fs;
fs.RecodringTime.Parse(p + 16);
// fs.InterchangeLevel = Get16(p + 18);
// fs.MaxInterchangeLevel = Get16(p + 20);
// fs.FileSetNumber = Get32(p + 40);
// fs.FileSetDescNumber = Get32(p + 44);
// fs.Id.Parse(p + 304);
// fs.CopyrightId.Parse(p + 336);
// fs.AbstractId.Parse(p + 368);
fs.RootDirICB.Parse(p + 400);
// fs.DomainId.Parse(p + 416);
// fs.SystemStreamDirICB.Parse(p + 464);
vol.FileSets.Add(fs);
// nextExtent.Parse(p + 448);
}
for (int fsIndex = 0; fsIndex < vol.FileSets.Size(); fsIndex++)
{
CFileSet &fs = vol.FileSets[fsIndex];
int fileIndex = Files.Size();
Files.Add(CFile());
RINOK(ReadFileItem(volIndex, fsIndex, fs.RootDirICB, kNumRecureseLevelsMax));
RINOK(FillRefs(fs, fileIndex, -1, kNumRecureseLevelsMax));
}
}
return S_OK;
}
HRESULT CInArchive::Open(IInStream *inStream, CProgressVirt *progress)
{
_progress = progress;
_stream = inStream;
HRESULT res;
try { res = Open2(); }
catch(...) { Clear(); res = S_FALSE; }
_stream.Release();
return res;
}
void CInArchive::Clear()
{
Partitions.Clear();
LogVols.Clear();
Items.Clear();
Files.Clear();
_fileNameLengthTotal = 0;
_numRefs = 0;
_numExtents = 0;
_inlineExtentsSize = 0;
_processedProgressBytes = 0;
}
UString CInArchive::GetComment() const
{
UString res;
for (int i = 0; i < LogVols.Size(); i++)
{
if (i > 0)
res += L" ";
res += LogVols[i].GetName();
}
return res;
}
static UString GetSpecName(const UString &name)
{
UString name2 = name;
name2.Trim();
if (name2.IsEmpty())
{
/*
wchar_t s[32];
ConvertUInt64ToString(id, s);
return L"[" + (UString)s + L"]";
*/
return L"[]";
}
return name;
}
static void UpdateWithName(UString &res, const UString &addString)
{
if (res.IsEmpty())
res = addString;
else
res = addString + WCHAR_PATH_SEPARATOR + res;
}
UString CInArchive::GetItemPath(int volIndex, int fsIndex, int refIndex,
bool showVolName, bool showFsName) const
{
// showVolName = true;
const CLogVol &vol = LogVols[volIndex];
const CFileSet &fs = vol.FileSets[fsIndex];
UString name;
for (;;)
{
const CRef &ref = fs.Refs[refIndex];
refIndex = ref.Parent;
if (refIndex < 0)
break;
UpdateWithName(name, GetSpecName(Files[ref.FileIndex].GetName()));
}
if (showFsName)
{
wchar_t s[32];
ConvertUInt64ToString(fsIndex, s);
UString newName = L"File Set ";
newName += s;
UpdateWithName(name, newName);
}
if (showVolName)
{
wchar_t s[32];
ConvertUInt64ToString(volIndex, s);
UString newName = s;
UString newName2 = vol.GetName();
if (newName2.IsEmpty())
newName2 = L"Volume";
newName += L'-';
newName += newName2;
UpdateWithName(name, newName);
}
return name;
}
}}
?? 快捷鍵說明
復制代碼
Ctrl + C
搜索代碼
Ctrl + F
全屏模式
F11
切換主題
Ctrl + Shift + D
顯示快捷鍵
?
增大字號
Ctrl + =
減小字號
Ctrl + -