Mercurial > vba-linux
view src/win32/7zip/7z/CPP/7zip/Archive/7z/7zExtract.cpp @ 1:f9f4f1b99eed
importing src directory
author | Robert McIntyre <rlm@mit.edu> |
---|---|
date | Sat, 03 Mar 2012 10:31:27 -0600 |
parents | |
children |
line wrap: on
line source
1 // 7zExtract.cpp3 #include "StdAfx.h"5 #include "7zHandler.h"6 #include "7zFolderOutStream.h"7 #include "7zDecode.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"15 namespace NArchive {16 namespace N7z {18 struct CExtractFolderInfo19 {20 #ifdef _7Z_VOL21 int VolumeIndex;22 #endif23 CNum FileIndex;24 CNum FolderIndex;25 CBoolVector ExtractStatuses;26 UInt64 UnpackSize;27 CExtractFolderInfo(28 #ifdef _7Z_VOL29 int volumeIndex,30 #endif31 CNum fileIndex, CNum folderIndex):32 #ifdef _7Z_VOL33 VolumeIndex(volumeIndex),34 #endif35 FileIndex(fileIndex),36 FolderIndex(folderIndex),37 UnpackSize(0)38 {39 if (fileIndex != kNumNoIndex)40 {41 ExtractStatuses.Reserve(1);42 ExtractStatuses.Add(true);43 }44 };45 };47 STDMETHODIMP CHandler::Extract(const UInt32* indices, UInt32 numItems,48 Int32 testModeSpec, IArchiveExtractCallback *extractCallbackSpec)49 {50 COM_TRY_BEGIN51 bool testMode = (testModeSpec != 0);52 CMyComPtr<IArchiveExtractCallback> extractCallback = extractCallbackSpec;53 UInt64 importantTotalUnpacked = 0;55 bool allFilesMode = (numItems == UInt32(-1));56 if (allFilesMode)57 numItems =58 #ifdef _7Z_VOL59 _refs.Size();60 #else61 _db.Files.Size();62 #endif64 if(numItems == 0)65 return S_OK;67 /*68 if(_volumes.Size() != 1)69 return E_FAIL;70 const CVolume &volume = _volumes.Front();71 const CArchiveDatabaseEx &_db = volume.Database;72 IInStream *_inStream = volume.Stream;73 */75 CObjectVector<CExtractFolderInfo> extractFolderInfoVector;76 for(UInt32 ii = 0; ii < numItems; ii++)77 {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++)83 {84 #ifdef _7Z_VOL85 // 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 &db = volume.Database;91 UInt32 fileIndex = ref.ItemIndex;92 #else93 const CArchiveDatabaseEx &db = _db;94 UInt32 fileIndex = ref2Index;95 #endif97 CNum folderIndex = db.FileIndexToFolderIndexMap[fileIndex];98 if (folderIndex == kNumNoIndex)99 {100 extractFolderInfoVector.Add(CExtractFolderInfo(101 #ifdef _7Z_VOL102 volumeIndex,103 #endif104 fileIndex, kNumNoIndex));105 continue;106 }107 if (extractFolderInfoVector.IsEmpty() ||108 folderIndex != extractFolderInfoVector.Back().FolderIndex109 #ifdef _7Z_VOL110 || volumeIndex != extractFolderInfoVector.Back().VolumeIndex111 #endif112 )113 {114 extractFolderInfoVector.Add(CExtractFolderInfo(115 #ifdef _7Z_VOL116 volumeIndex,117 #endif118 kNumNoIndex, folderIndex));119 const CFolder &folderInfo = db.Folders[folderIndex];120 UInt64 unpackSize = folderInfo.GetUnpackSize();121 importantTotalUnpacked += unpackSize;122 extractFolderInfoVector.Back().UnpackSize = unpackSize;123 }125 CExtractFolderInfo &efi = extractFolderInfoVector.Back();127 // const CFolderInfo &folderInfo = m_dam_Folders[folderIndex];128 CNum startIndex = db.FolderStartFileIndex[folderIndex];129 for (CNum index = efi.ExtractStatuses.Size();130 index <= fileIndex - startIndex; index++)131 {132 // UInt64 unpackSize = _db.Files[startIndex + index].UnpackSize;133 // Count partial_folder_size134 // efi.UnpackSize += unpackSize;135 // importantTotalUnpacked += unpackSize;136 efi.ExtractStatuses.Add(index == fileIndex - startIndex);137 }138 }139 }141 extractCallback->SetTotal(importantTotalUnpacked);143 CDecoder decoder(144 #ifdef _ST_MODE145 false146 #else147 true148 #endif149 );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)164 {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);177 #ifdef _7Z_VOL178 const CVolume &volume = _volumes[efi.VolumeIndex];179 const CArchiveDatabaseEx &db = volume.Database;180 #else181 const CArchiveDatabaseEx &db = _db;182 #endif184 CNum startIndex;185 if (efi.FileIndex != kNumNoIndex)186 startIndex = efi.FileIndex;187 else188 startIndex = db.FolderStartFileIndex[efi.FolderIndex];191 HRESULT result = folderOutStream->Init(&db,192 #ifdef _7Z_VOL193 volume.StartRef2Index,194 #else195 0,196 #endif197 startIndex,198 &efi.ExtractStatuses, extractCallback, testMode, _crcSize != 0);200 RINOK(result);202 if (efi.FileIndex != kNumNoIndex)203 continue;205 CNum folderIndex = efi.FolderIndex;206 const CFolder &folderInfo = db.Folders[folderIndex];208 totalFolderPacked = _db.GetFolderFullPackSize(folderIndex);210 CNum packStreamIndex = db.FolderStartPackStreamIndex[folderIndex];211 UInt64 folderStartPackPos = db.GetFolderStreamPos(folderIndex, 0);213 #ifndef _NO_CRYPTO214 CMyComPtr<ICryptoGetTextPassword> getTextPassword;215 if (extractCallback)216 extractCallback.QueryInterface(IID_ICryptoGetTextPassword, &getTextPassword);217 #endif219 try220 {221 #ifndef _NO_CRYPTO222 bool passwordIsDefined;223 #endif225 HRESULT result = decoder.Decode(226 EXTERNAL_CODECS_VARS227 #ifdef _7Z_VOL228 volume.Stream,229 #else230 _inStream,231 #endif232 folderStartPackPos,233 &db.PackSizes[packStreamIndex],234 folderInfo,235 outStream,236 progress237 #ifndef _NO_CRYPTO238 , getTextPassword, passwordIsDefined239 #endif240 #ifdef COMPRESS_MT241 , true, _numThreads242 #endif243 );245 if (result == S_FALSE)246 {247 RINOK(folderOutStream->FlushCorrupted(NArchive::NExtract::NOperationResult::kDataError));248 continue;249 }250 if (result == E_NOTIMPL)251 {252 RINOK(folderOutStream->FlushCorrupted(NArchive::NExtract::NOperationResult::kUnSupportedMethod));253 continue;254 }255 if (result != S_OK)256 return result;257 if (folderOutStream->WasWritingFinished() != S_OK)258 {259 RINOK(folderOutStream->FlushCorrupted(NArchive::NExtract::NOperationResult::kDataError));260 continue;261 }262 }263 catch(...)264 {265 RINOK(folderOutStream->FlushCorrupted(NArchive::NExtract::NOperationResult::kDataError));266 continue;267 }268 }269 return S_OK;270 COM_TRY_END271 }273 }}