6 #include "7zFolderOutStream.h"
8 // #include "7z1Decode.h"
10 #include "../../../Common/ComTry.h"
11 #include "../../Common/StreamObjects.h"
12 #include "../../Common/ProgressUtils.h"
13 #include "../../Common/LimitedStreams.h"
18 struct CExtractFolderInfo
25 CBoolVector ExtractStatuses;
31 CNum fileIndex, CNum folderIndex):
33 VolumeIndex(volumeIndex),
36 FolderIndex(folderIndex),
39 if (fileIndex != kNumNoIndex)
41 ExtractStatuses.Reserve(1);
42 ExtractStatuses.Add(true);
47 STDMETHODIMP CHandler::Extract(const UInt32* indices, UInt32 numItems,
48 Int32 testModeSpec, IArchiveExtractCallback *extractCallbackSpec)
51 bool testMode = (testModeSpec != 0);
52 CMyComPtr<IArchiveExtractCallback> extractCallback = extractCallbackSpec;
53 UInt64 importantTotalUnPacked = 0;
55 bool allFilesMode = (numItems == UInt32(-1));
61 _database.Files.Size();
68 if(_volumes.Size() != 1)
70 const CVolume &volume = _volumes.Front();
71 const CArchiveDatabaseEx &_database = volume.Database;
72 IInStream *_inStream = volume.Stream;
75 CObjectVector<CExtractFolderInfo> extractFolderInfoVector;
76 for(UInt32 ii = 0; ii < numItems; ii++)
78 // UInt32 fileIndex = allFilesMode ? indexIndex : indices[indexIndex];
79 UInt32 ref2Index = allFilesMode ? ii : indices[ii];
80 // const CRef2 &ref2 = _refs[ref2Index];
82 // for(UInt32 ri = 0; ri < ref2.Refs.Size(); ri++)
85 // const CRef &ref = ref2.Refs[ri];
86 const CRef &ref = _refs[ref2Index];
88 int volumeIndex = ref.VolumeIndex;
89 const CVolume &volume = _volumes[volumeIndex];
90 const CArchiveDatabaseEx &database = volume.Database;
91 UInt32 fileIndex = ref.ItemIndex;
93 const CArchiveDatabaseEx &database = _database;
94 UInt32 fileIndex = ref2Index;
97 CNum folderIndex = database.FileIndexToFolderIndexMap[fileIndex];
98 if (folderIndex == kNumNoIndex)
100 extractFolderInfoVector.Add(CExtractFolderInfo(
104 fileIndex, kNumNoIndex));
107 if (extractFolderInfoVector.IsEmpty() ||
108 folderIndex != extractFolderInfoVector.Back().FolderIndex
110 || volumeIndex != extractFolderInfoVector.Back().VolumeIndex
114 extractFolderInfoVector.Add(CExtractFolderInfo(
118 kNumNoIndex, folderIndex));
119 const CFolder &folderInfo = database.Folders[folderIndex];
120 UInt64 unPackSize = folderInfo.GetUnPackSize();
121 importantTotalUnPacked += unPackSize;
122 extractFolderInfoVector.Back().UnPackSize = unPackSize;
125 CExtractFolderInfo &efi = extractFolderInfoVector.Back();
127 // const CFolderInfo &folderInfo = m_dam_Folders[folderIndex];
128 CNum startIndex = database.FolderStartFileIndex[folderIndex];
129 for (CNum index = efi.ExtractStatuses.Size();
130 index <= fileIndex - startIndex; index++)
132 // UInt64 unPackSize = _database.Files[startIndex + index].UnPackSize;
133 // Count partial_folder_size
134 // efi.UnPackSize += unPackSize;
135 // importantTotalUnPacked += unPackSize;
136 efi.ExtractStatuses.Add(index == fileIndex - startIndex);
141 extractCallback->SetTotal(importantTotalUnPacked);
150 // CDecoder1 decoder;
152 UInt64 currentTotalPacked = 0;
153 UInt64 currentTotalUnPacked = 0;
154 UInt64 totalFolderUnPacked;
155 UInt64 totalFolderPacked;
157 CLocalProgress *lps = new CLocalProgress;
158 CMyComPtr<ICompressProgressInfo> progress = lps;
159 lps->Init(extractCallback, false);
161 for(int i = 0; i < extractFolderInfoVector.Size(); i++,
162 currentTotalUnPacked += totalFolderUnPacked,
163 currentTotalPacked += totalFolderPacked)
165 lps->OutSize = currentTotalUnPacked;
166 lps->InSize = currentTotalPacked;
167 RINOK(lps->SetCur());
169 const CExtractFolderInfo &efi = extractFolderInfoVector[i];
170 totalFolderUnPacked = efi.UnPackSize;
172 totalFolderPacked = 0;
174 CFolderOutStream *folderOutStream = new CFolderOutStream;
175 CMyComPtr<ISequentialOutStream> outStream(folderOutStream);
178 const CVolume &volume = _volumes[efi.VolumeIndex];
179 const CArchiveDatabaseEx &database = volume.Database;
181 const CArchiveDatabaseEx &database = _database;
185 if (efi.FileIndex != kNumNoIndex)
186 startIndex = efi.FileIndex;
188 startIndex = database.FolderStartFileIndex[efi.FolderIndex];
191 HRESULT result = folderOutStream->Init(&database,
193 volume.StartRef2Index,
198 &efi.ExtractStatuses, extractCallback, testMode, _crcSize != 0);
202 if (efi.FileIndex != kNumNoIndex)
205 CNum folderIndex = efi.FolderIndex;
206 const CFolder &folderInfo = database.Folders[folderIndex];
208 totalFolderPacked = _database.GetFolderFullPackSize(folderIndex);
210 CNum packStreamIndex = database.FolderStartPackStreamIndex[folderIndex];
211 UInt64 folderStartPackPos = database.GetFolderStreamPos(folderIndex, 0);
214 CMyComPtr<ICryptoGetTextPassword> getTextPassword;
216 extractCallback.QueryInterface(IID_ICryptoGetTextPassword, &getTextPassword);
221 HRESULT result = decoder.Decode(
229 &database.PackSizes[packStreamIndex],
241 if (result == S_FALSE)
243 RINOK(folderOutStream->FlushCorrupted(NArchive::NExtract::NOperationResult::kDataError));
246 if (result == E_NOTIMPL)
248 RINOK(folderOutStream->FlushCorrupted(NArchive::NExtract::NOperationResult::kUnSupportedMethod));
253 if (folderOutStream->WasWritingFinished() != S_OK)
255 RINOK(folderOutStream->FlushCorrupted(NArchive::NExtract::NOperationResult::kDataError));
261 RINOK(folderOutStream->FlushCorrupted(NArchive::NExtract::NOperationResult::kDataError));