| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279 | /* * unrar.cpp * --------- * Purpose: Implementation file for extracting modules from .rar archives * Notes  : (currently none) * Authors: OpenMPT Devs * The OpenMPT source code is released under the BSD license. Read LICENSE for more details. */#include "stdafx.h"#include "unrar.h"#ifdef MPT_WITH_UNRAR#include "../common/mptFileIO.h"#if MPT_OS_WINDOWS #include <windows.h>#else // !MPT_OS_WINDOWS #ifdef _UNIX  #define MPT_UNRAR_UNIX_WAS_DEFINED #else  #define _UNIX #endif#endif // MPT_OS_WINDOWS#include "unrar/dll.hpp"#if !MPT_OS_WINDOWS #ifndef MPT_UNRAR_UNIX_WAS_DEFINED  #undef _UNIX  #undef MPT_UNRAR_UNIX_WAS_DEFINED #endif#endif // !MPT_OS_WINDOWS#endif // MPT_WITH_UNRAROPENMPT_NAMESPACE_BEGIN#ifdef MPT_WITH_UNRARstruct RARHandle // RAII{	HANDLE rar = nullptr;	explicit RARHandle(HANDLE rar_) : rar(rar_) { return; }	RARHandle(const RARHandle &) = delete;	~RARHandle() { if(rar) RARCloseArchive(rar); }	operator HANDLE () const { return rar; }};static int CALLBACK RARCallback(unsigned int msg, LPARAM userData, LPARAM p1, LPARAM p2){	int result = 0;	CRarArchive *that = reinterpret_cast<CRarArchive *>(userData);	switch(msg)	{	case UCM_PROCESSDATA:		// Receive extracted data		that->RARCallbackProcessData(reinterpret_cast<const char *>(p1), p2);		result = 1;		break;	default:		// No support for passwords or volumes		result = -1;		break;	}	return result;}void CRarArchive::RARCallbackProcessData(const char * buf, std::size_t size){	if(!captureCurrentFile)	{		return;	}	mpt::append(data, buf, buf + size);}CRarArchive::CRarArchive(FileReader &file)	: ArchiveBase(file){	// NOTE:	//  We open the archive twice, once for listing the contents in the	// constructor and once for actual decompression in ExtractFile.	//  For solid archives, listing the contents via RAR_OM_LIST is way faster.	// The overhead of opening twice for non-solid archives is negligable if the	// archive does not contain a lot of files (and archives with large amount of	// files are pretty useless for OpenMPT anyway).	// Early reject files with no Rar! magic	// so that we do not have to instantiate OnDiskFileWrapper.	inFile.Rewind();	if(!inFile.ReadMagic("Rar!\x1A"))	{		return;	}	inFile.Rewind();	diskFile = std::make_unique<OnDiskFileWrapper>(inFile);	if(!diskFile->IsValid())	{		return;	}	std::wstring ArcName = diskFile->GetFilename().ToWide();	std::vector<wchar_t> ArcNameBuf(ArcName.c_str(), ArcName.c_str() + ArcName.length() + 1);	std::vector<wchar_t> CmtBuf(65536);	RAROpenArchiveDataEx ArchiveData = {};	ArchiveData.OpenMode = RAR_OM_LIST;	ArchiveData.ArcNameW = ArcNameBuf.data();	ArchiveData.CmtBufW = CmtBuf.data();	ArchiveData.CmtBufSize = static_cast<unsigned int>(CmtBuf.size());	RARHandle rar(RAROpenArchiveEx(&ArchiveData));	if(!rar)	{		Reset();		return;	}	switch(ArchiveData.CmtState)	{	case 1:		if(ArchiveData.CmtSize)		{			comment = mpt::ToUnicode(std::wstring(ArchiveData.CmtBufW, ArchiveData.CmtBufW + ArchiveData.CmtSize - 1));			break;		}		[[fallthrough]];	case 0:		comment = mpt::ustring();		break;	default:		Reset();		return;		break;	}	bool eof = false;	int RARResult = 0;	while(!eof)	{		RARHeaderDataEx HeaderData = {};		RARResult = RARReadHeaderEx(rar, &HeaderData);		switch(RARResult)		{		case ERAR_SUCCESS:			break;		case ERAR_END_ARCHIVE:			eof = true;			continue;			break;		default:			Reset();			return;			break;		}		ArchiveFileInfo fileInfo;		fileInfo.name = mpt::PathString::FromWide(HeaderData.FileNameW);		fileInfo.type = ArchiveFileType::Normal;		fileInfo.size = HeaderData.UnpSize;		contents.push_back(fileInfo);		RARResult = RARProcessFileW(rar, RAR_SKIP, NULL, NULL);		switch(RARResult)		{		case ERAR_SUCCESS:			break;		default:			Reset();			return;			break;		}	}}CRarArchive::~CRarArchive(){}bool CRarArchive::ExtractFile(std::size_t index){	if(!diskFile || !diskFile->IsValid())	{		return false;	}	if(index >= contents.size())	{		return false;	}	std::wstring ArcName = diskFile->GetFilename().ToWide();	std::vector<wchar_t> ArcNameBuf(ArcName.c_str(), ArcName.c_str() + ArcName.length() + 1);	RAROpenArchiveDataEx ArchiveData = {};	ArchiveData.OpenMode = RAR_OM_EXTRACT;	ArchiveData.ArcNameW = ArcNameBuf.data();	ArchiveData.Callback = RARCallback;	ArchiveData.UserData = reinterpret_cast<LPARAM>(this);	RARHandle rar(RAROpenArchiveEx(&ArchiveData));	if(!rar)	{		ResetFile();		return false;	}	std::size_t i = 0;	int RARResult = 0;	bool eof = false;	while(!eof)	{		RARHeaderDataEx HeaderData = {};		RARResult = RARReadHeaderEx(rar, &HeaderData);		switch(RARResult)		{		case ERAR_SUCCESS:			break;		case ERAR_END_ARCHIVE:			eof = true;			continue;			break;		default:			ResetFile();			return false;			break;		}		captureCurrentFile = (i == index);		RARResult = RARProcessFileW(rar, captureCurrentFile ? RAR_TEST : RAR_SKIP, NULL, NULL);		switch(RARResult)		{		case ERAR_SUCCESS:			break;		default:			ResetFile();			return false;			break;		}		if(captureCurrentFile)		{ // done			return true;		}		captureCurrentFile = false;		++i;	}	return false;}void CRarArchive::Reset(){	captureCurrentFile = false;	comment = mpt::ustring();	contents = std::vector<ArchiveFileInfo>();	data = std::vector<char>();}void CRarArchive::ResetFile(){	captureCurrentFile = false;	data = std::vector<char>();}#endif // MPT_WITH_UNRAROPENMPT_NAMESPACE_END
 |