mirror of
https://github.com/ZDoom/Raze.git
synced 2025-01-10 02:50:49 +00:00
a15ac43722
Some mods provide pregenerated resources, this allows loading them without picking them apart first.
381 lines
8.7 KiB
C++
381 lines
8.7 KiB
C++
/*
|
|
** file_7z.cpp
|
|
**
|
|
**---------------------------------------------------------------------------
|
|
** Copyright 2009 Randy Heit
|
|
** Copyright 2009 Christoph Oelckers
|
|
** All rights reserved.
|
|
**
|
|
** Redistribution and use in source and binary forms, with or without
|
|
** modification, are permitted provided that the following conditions
|
|
** are met:
|
|
**
|
|
** 1. Redistributions of source code must retain the above copyright
|
|
** notice, this list of conditions and the following disclaimer.
|
|
** 2. Redistributions in binary form must reproduce the above copyright
|
|
** notice, this list of conditions and the following disclaimer in the
|
|
** documentation and/or other materials provided with the distribution.
|
|
** 3. The name of the author may not be used to endorse or promote products
|
|
** derived from this software without specific prior written permission.
|
|
**
|
|
** THIS SOFTWARE IS PROVIDED BY THE AUTHOR ``AS IS'' AND ANY EXPRESS OR
|
|
** IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES
|
|
** OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED.
|
|
** IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
** INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT
|
|
** NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
|
|
** DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
|
|
** THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
|
|
** (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
|
|
** THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
**---------------------------------------------------------------------------
|
|
**
|
|
**
|
|
*/
|
|
|
|
// Note that 7z made the unwise decision to include windows.h :(
|
|
#include "7z.h"
|
|
#include "7zCrc.h"
|
|
|
|
#include "resourcefile.h"
|
|
#include "cmdlib.h"
|
|
#include "printf.h"
|
|
|
|
|
|
|
|
//-----------------------------------------------------------------------
|
|
//
|
|
// Interface classes to 7z library
|
|
//
|
|
//-----------------------------------------------------------------------
|
|
|
|
extern ISzAlloc g_Alloc;
|
|
|
|
struct CZDFileInStream
|
|
{
|
|
ISeekInStream s;
|
|
FileReader &File;
|
|
|
|
CZDFileInStream(FileReader &_file)
|
|
: File(_file)
|
|
{
|
|
s.Read = Read;
|
|
s.Seek = Seek;
|
|
}
|
|
|
|
static SRes Read(const ISeekInStream *pp, void *buf, size_t *size)
|
|
{
|
|
CZDFileInStream *p = (CZDFileInStream *)pp;
|
|
auto numread = p->File.Read(buf, (long)*size);
|
|
if (numread < 0)
|
|
{
|
|
*size = 0;
|
|
return SZ_ERROR_READ;
|
|
}
|
|
*size = numread;
|
|
return SZ_OK;
|
|
}
|
|
|
|
static SRes Seek(const ISeekInStream *pp, Int64 *pos, ESzSeek origin)
|
|
{
|
|
CZDFileInStream *p = (CZDFileInStream *)pp;
|
|
FileReader::ESeek move_method;
|
|
int res;
|
|
if (origin == SZ_SEEK_SET)
|
|
{
|
|
move_method = FileReader::SeekSet;
|
|
}
|
|
else if (origin == SZ_SEEK_CUR)
|
|
{
|
|
move_method = FileReader::SeekCur;
|
|
}
|
|
else if (origin == SZ_SEEK_END)
|
|
{
|
|
move_method = FileReader::SeekEnd;
|
|
}
|
|
else
|
|
{
|
|
return 1;
|
|
}
|
|
res = (int)p->File.Seek((long)*pos, move_method);
|
|
*pos = p->File.Tell();
|
|
return res;
|
|
}
|
|
};
|
|
|
|
struct C7zArchive
|
|
{
|
|
CSzArEx DB;
|
|
CZDFileInStream ArchiveStream;
|
|
CLookToRead2 LookStream;
|
|
Byte StreamBuffer[1<<14];
|
|
UInt32 BlockIndex;
|
|
Byte *OutBuffer;
|
|
size_t OutBufferSize;
|
|
|
|
C7zArchive(FileReader &file) : ArchiveStream(file)
|
|
{
|
|
if (g_CrcTable[1] == 0)
|
|
{
|
|
CrcGenerateTable();
|
|
}
|
|
file.Seek(0, FileReader::SeekSet);
|
|
LookToRead2_CreateVTable(&LookStream, false);
|
|
LookStream.realStream = &ArchiveStream.s;
|
|
LookToRead2_Init(&LookStream);
|
|
LookStream.bufSize = sizeof(StreamBuffer);
|
|
LookStream.buf = StreamBuffer;
|
|
SzArEx_Init(&DB);
|
|
BlockIndex = 0xFFFFFFFF;
|
|
OutBuffer = NULL;
|
|
OutBufferSize = 0;
|
|
}
|
|
|
|
~C7zArchive()
|
|
{
|
|
if (OutBuffer != NULL)
|
|
{
|
|
IAlloc_Free(&g_Alloc, OutBuffer);
|
|
}
|
|
SzArEx_Free(&DB, &g_Alloc);
|
|
}
|
|
|
|
SRes Open()
|
|
{
|
|
return SzArEx_Open(&DB, &LookStream.vt, &g_Alloc, &g_Alloc);
|
|
}
|
|
|
|
SRes Extract(UInt32 file_index, char *buffer)
|
|
{
|
|
size_t offset, out_size_processed;
|
|
SRes res = SzArEx_Extract(&DB, &LookStream.vt, file_index,
|
|
&BlockIndex, &OutBuffer, &OutBufferSize,
|
|
&offset, &out_size_processed,
|
|
&g_Alloc, &g_Alloc);
|
|
if (res == SZ_OK)
|
|
{
|
|
memcpy(buffer, OutBuffer + offset, out_size_processed);
|
|
}
|
|
return res;
|
|
}
|
|
};
|
|
//==========================================================================
|
|
//
|
|
// Zip Lump
|
|
//
|
|
//==========================================================================
|
|
|
|
struct F7ZLump : public FResourceLump
|
|
{
|
|
int Position;
|
|
|
|
virtual int FillCache();
|
|
|
|
};
|
|
|
|
|
|
//==========================================================================
|
|
//
|
|
// 7-zip file
|
|
//
|
|
//==========================================================================
|
|
|
|
class F7ZFile : public FResourceFile
|
|
{
|
|
friend struct F7ZLump;
|
|
|
|
F7ZLump *Lumps;
|
|
C7zArchive *Archive;
|
|
|
|
public:
|
|
F7ZFile(const char * filename, FileReader &filer);
|
|
bool Open(bool quiet, LumpFilterInfo* filter);
|
|
virtual ~F7ZFile();
|
|
virtual FResourceLump *GetLump(int no) { return ((unsigned)no < NumLumps)? &Lumps[no] : NULL; }
|
|
};
|
|
|
|
|
|
|
|
//==========================================================================
|
|
//
|
|
// 7Z file
|
|
//
|
|
//==========================================================================
|
|
|
|
F7ZFile::F7ZFile(const char * filename, FileReader &filer)
|
|
: FResourceFile(filename, filer)
|
|
{
|
|
Lumps = NULL;
|
|
Archive = NULL;
|
|
}
|
|
|
|
|
|
//==========================================================================
|
|
//
|
|
// Open it
|
|
//
|
|
//==========================================================================
|
|
|
|
bool F7ZFile::Open(bool quiet, LumpFilterInfo *filter)
|
|
{
|
|
Archive = new C7zArchive(Reader);
|
|
int skipped = 0;
|
|
SRes res;
|
|
|
|
res = Archive->Open();
|
|
if (res != SZ_OK)
|
|
{
|
|
delete Archive;
|
|
Archive = NULL;
|
|
if (!quiet)
|
|
{
|
|
Printf("\n" TEXTCOLOR_RED "%s: ", FileName.GetChars());
|
|
if (res == SZ_ERROR_UNSUPPORTED)
|
|
{
|
|
Printf("Decoder does not support this archive\n");
|
|
}
|
|
else if (res == SZ_ERROR_MEM)
|
|
{
|
|
Printf("Cannot allocate memory\n");
|
|
}
|
|
else if (res == SZ_ERROR_CRC)
|
|
{
|
|
Printf("CRC error\n");
|
|
}
|
|
else
|
|
{
|
|
Printf("error #%d\n", res);
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
CSzArEx* const archPtr = &Archive->DB;
|
|
|
|
NumLumps = archPtr->NumFiles;
|
|
Lumps = new F7ZLump[NumLumps];
|
|
|
|
F7ZLump *lump_p = Lumps;
|
|
TArray<UInt16> nameUTF16;
|
|
TArray<char> nameASCII;
|
|
|
|
for (uint32_t i = 0; i < NumLumps; ++i)
|
|
{
|
|
// skip Directories
|
|
if (SzArEx_IsDir(archPtr, i))
|
|
{
|
|
skipped++;
|
|
continue;
|
|
}
|
|
|
|
const size_t nameLength = SzArEx_GetFileNameUtf16(archPtr, i, NULL);
|
|
|
|
if (0 == nameLength)
|
|
{
|
|
++skipped;
|
|
continue;
|
|
}
|
|
|
|
nameUTF16.Resize((unsigned)nameLength);
|
|
nameASCII.Resize((unsigned)nameLength);
|
|
SzArEx_GetFileNameUtf16(archPtr, i, &nameUTF16[0]);
|
|
for (size_t c = 0; c < nameLength; ++c)
|
|
{
|
|
nameASCII[c] = static_cast<char>(nameUTF16[c]);
|
|
}
|
|
FixPathSeperator(&nameASCII[0]);
|
|
|
|
FString name = &nameASCII[0];
|
|
name.ToLower();
|
|
|
|
lump_p->LumpNameSetup(name);
|
|
lump_p->LumpSize = static_cast<int>(SzArEx_GetFileSize(archPtr, i));
|
|
lump_p->Owner = this;
|
|
lump_p->Flags = LUMPF_FULLPATH|LUMPF_COMPRESSED;
|
|
lump_p->Position = i;
|
|
lump_p->CheckEmbedded(filter);
|
|
lump_p++;
|
|
}
|
|
// Resize the lump record array to its actual size
|
|
NumLumps -= skipped;
|
|
|
|
if (NumLumps > 0)
|
|
{
|
|
// Quick check for unsupported compression method
|
|
|
|
TArray<char> temp;
|
|
temp.Resize(Lumps[0].LumpSize);
|
|
|
|
if (SZ_OK != Archive->Extract(Lumps[0].Position, &temp[0]))
|
|
{
|
|
if (!quiet) Printf("\n%s: unsupported 7z/LZMA file!\n", FileName.GetChars());
|
|
return false;
|
|
}
|
|
}
|
|
|
|
GenerateHash();
|
|
PostProcessArchive(&Lumps[0], sizeof(F7ZLump), filter);
|
|
return true;
|
|
}
|
|
|
|
//==========================================================================
|
|
//
|
|
//
|
|
//
|
|
//==========================================================================
|
|
|
|
F7ZFile::~F7ZFile()
|
|
{
|
|
if (Lumps != NULL)
|
|
{
|
|
delete[] Lumps;
|
|
}
|
|
if (Archive != NULL)
|
|
{
|
|
delete Archive;
|
|
}
|
|
}
|
|
|
|
//==========================================================================
|
|
//
|
|
// Fills the lump cache and performs decompression
|
|
//
|
|
//==========================================================================
|
|
|
|
int F7ZLump::FillCache()
|
|
{
|
|
Cache = new char[LumpSize];
|
|
static_cast<F7ZFile*>(Owner)->Archive->Extract(Position, Cache);
|
|
RefCount = 1;
|
|
return 1;
|
|
}
|
|
|
|
//==========================================================================
|
|
//
|
|
// File open
|
|
//
|
|
//==========================================================================
|
|
|
|
FResourceFile *Check7Z(const char *filename, FileReader &file, bool quiet, LumpFilterInfo* filter)
|
|
{
|
|
char head[k7zSignatureSize];
|
|
|
|
if (file.GetLength() >= k7zSignatureSize)
|
|
{
|
|
file.Seek(0, FileReader::SeekSet);
|
|
file.Read(&head, k7zSignatureSize);
|
|
file.Seek(0, FileReader::SeekSet);
|
|
if (!memcmp(head, k7zSignature, k7zSignatureSize))
|
|
{
|
|
auto rf = new F7ZFile(filename, file);
|
|
if (rf->Open(quiet, filter)) return rf;
|
|
|
|
file = std::move(rf->Reader); // to avoid destruction of reader
|
|
delete rf;
|
|
}
|
|
}
|
|
return NULL;
|
|
}
|
|
|
|
|
|
|