📄 zipfile.cpp
字号:
//
// ZIPFile.cpp
//
// Copyright (c) Shareaza Development Team, 2002-2004.
// This file is part of SHAREAZA (www.shareaza.com)
//
// Shareaza is free software; you can redistribute it
// and/or modify it under the terms of the GNU General Public License
// as published by the Free Software Foundation; either version 2 of
// the License, or (at your option) any later version.
//
// Shareaza is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with Shareaza; if not, write to the Free Software
// Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
//
#include "StdAfx.h"
#include "ZIPFile.h"
#include <zlib/zlib.h>
/////////////////////////////////////////////////////////////////////////////
// CZIPFile construction
CZIPFile::CZIPFile(HANDLE hAttach)
{
m_bAttach = FALSE;
m_hFile = INVALID_HANDLE_VALUE;
m_pFile = NULL;
m_nFile = 0;
if ( hAttach != INVALID_HANDLE_VALUE ) Attach( hAttach );
}
CZIPFile::~CZIPFile()
{
Close();
}
/////////////////////////////////////////////////////////////////////////////
// CZIPFile open
BOOL CZIPFile::Open(LPCTSTR pszFile)
{
ASSERT( pszFile != NULL );
Close();
m_bAttach = FALSE;
m_hFile = CreateFile( pszFile, GENERIC_READ, FILE_SHARE_READ, NULL,
OPEN_EXISTING, FILE_ATTRIBUTE_NORMAL, NULL );
if ( m_hFile == INVALID_HANDLE_VALUE ) return FALSE;
if ( LocateCentralDirectory() )
{
return TRUE;
}
else
{
Close();
return FALSE;
}
}
/////////////////////////////////////////////////////////////////////////////
// CZIPFile attach
BOOL CZIPFile::Attach(HANDLE hFile)
{
ASSERT( hFile != INVALID_HANDLE_VALUE );
Close();
m_bAttach = TRUE;
m_hFile = hFile;
if ( LocateCentralDirectory() )
{
return TRUE;
}
else
{
Close();
return FALSE;
}
}
/////////////////////////////////////////////////////////////////////////////
// CZIPFile open test
BOOL CZIPFile::IsOpen() const
{
return m_hFile != INVALID_HANDLE_VALUE;
}
/////////////////////////////////////////////////////////////////////////////
// CZIPFile close
void CZIPFile::Close()
{
if ( m_hFile != INVALID_HANDLE_VALUE )
{
if ( ! m_bAttach ) CloseHandle( m_hFile );
m_hFile = INVALID_HANDLE_VALUE;
}
if ( m_pFile != NULL ) delete [] m_pFile;
m_pFile = NULL;
m_nFile = 0;
}
/////////////////////////////////////////////////////////////////////////////
// CZIPFile get the file count
int CZIPFile::GetCount() const
{
return m_nFile;
}
/////////////////////////////////////////////////////////////////////////////
// CZIPFile get a particular file
CZIPFile::File* CZIPFile::GetFile(int nFile) const
{
return ( nFile < 0 || nFile >= m_nFile ) ? NULL : m_pFile + nFile;
}
/////////////////////////////////////////////////////////////////////////////
// CZIPFile lookup a file by name
CZIPFile::File* CZIPFile::GetFile(LPCTSTR pszFile, BOOL bPartial) const
{
File* pFile = m_pFile;
for ( int nFile = m_nFile ; nFile ; nFile--, pFile++ )
{
if ( bPartial )
{
LPCTSTR pszName = _tcsrchr( pFile->m_sName, '/' );
pszName = pszName ? pszName + 1 : (LPCTSTR)pFile->m_sName;
if ( _tcsicoll( pszName, pszFile ) == 0 ) return pFile;
}
else
{
if ( _tcsicoll( pFile->m_sName, pszFile ) == 0 ) return pFile;
}
}
return NULL;
}
/////////////////////////////////////////////////////////////////////////////
// CZIPFile locate the central directory
#pragma pack(1)
typedef struct
{
DWORD nSignature; // 0x06054b50
WORD nThisDisk;
WORD nDirectoryDisk;
WORD nFilesThisDisk;
WORD nTotalFiles;
DWORD nDirectorySize;
DWORD nDirectoryOffset;
WORD nCommentLen;
} ZIP_DIRECTORY_LOC;
#pragma pack()
BOOL CZIPFile::LocateCentralDirectory()
{
BYTE pBuffer[4096];
DWORD nBuffer = 0;
SetFilePointer( m_hFile, -4096, NULL, FILE_END );
if ( ! ReadFile( m_hFile, pBuffer, 4096, &nBuffer, NULL ) ) return FALSE;
if ( nBuffer < sizeof(ZIP_DIRECTORY_LOC) ) return FALSE;
ZIP_DIRECTORY_LOC* pLoc = NULL;
for ( DWORD nScan = 4 ; nScan < nBuffer ; nScan++ )
{
DWORD* pnSignature = (DWORD*)( pBuffer + nBuffer - nScan );
if ( *pnSignature == 0x06054b50 )
{
pLoc = (ZIP_DIRECTORY_LOC*)pnSignature;
break;
}
}
if ( pLoc == NULL ) return FALSE;
ASSERT( pLoc->nSignature == 0x06054b50 );
if ( GetFileSize( m_hFile, NULL ) < pLoc->nDirectorySize ) return FALSE;
if ( SetFilePointer( m_hFile, pLoc->nDirectoryOffset, NULL, FILE_BEGIN )
!= pLoc->nDirectoryOffset ) return FALSE;
BYTE* pDirectory = new BYTE[ pLoc->nDirectorySize ];
ReadFile( m_hFile, pDirectory, pLoc->nDirectorySize, &nBuffer, NULL );
if ( nBuffer == pLoc->nDirectorySize )
{
m_nFile = (int)pLoc->nTotalFiles;
m_pFile = new File[ m_nFile ];
if ( ! ParseCentralDirectory( pDirectory, pLoc->nDirectorySize ) )
{
delete [] m_pFile;
m_pFile = NULL;
m_nFile = 0;
}
}
delete [] pDirectory;
return ( m_nFile > 0 );
}
/////////////////////////////////////////////////////////////////////////////
// CZIPFile parse the central directory
#pragma pack(1)
typedef struct
{
DWORD nSignature; // 0x02014b50
WORD nWriteVersion;
WORD nReadVersion;
WORD nFlags;
WORD nCompression;
WORD nFileTime;
WORD nFileDate;
DWORD nCRC;
DWORD nCompressedSize;
DWORD nActualSize;
WORD nNameLen;
WORD nExtraLen;
WORD nCommentLen;
WORD nStartDisk;
WORD nInternalAttr;
DWORD nExternalAttr;
DWORD nLocalOffset;
} ZIP_CENTRAL_FILE;
#pragma pack()
BOOL CZIPFile::ParseCentralDirectory(BYTE* pDirectory, DWORD nDirectory)
{
for ( int nFile = 0 ; nFile < m_nFile ; nFile++ )
{
ZIP_CENTRAL_FILE* pRecord = (ZIP_CENTRAL_FILE*)pDirectory;
if ( nDirectory < sizeof(*pRecord) ) return FALSE;
if ( pRecord->nSignature != 0x02014b50 ) return FALSE;
pDirectory += sizeof(*pRecord);
nDirectory -= sizeof(*pRecord);
int nTailLen = (int)pRecord->nNameLen + (int)pRecord->nExtraLen + (int)pRecord->nCommentLen;
if ( nDirectory < (DWORD)nTailLen ) return FALSE;
m_pFile[ nFile ].m_pZIP = this;
m_pFile[ nFile ].m_nSize = pRecord->nActualSize;
m_pFile[ nFile ].m_nLocalOffset = pRecord->nLocalOffset;
m_pFile[ nFile ].m_nCompressedSize = pRecord->nCompressedSize;
m_pFile[ nFile ].m_nCompression = pRecord->nCompression;
LPTSTR pszName = m_pFile[ nFile ].m_sName.GetBuffer( pRecord->nNameLen );
for ( WORD nChar = 0 ; nChar < pRecord->nNameLen ; nChar++ )
{
pszName[ nChar ] = (TCHAR)pDirectory[ nChar ];
if ( pszName[ nChar ] == '\\' ) pszName[ nChar ] = '/';
}
m_pFile[ nFile ].m_sName.ReleaseBuffer( pRecord->nNameLen );
pDirectory += (DWORD)nTailLen;
nDirectory -= (DWORD)nTailLen;
}
return TRUE;
}
/////////////////////////////////////////////////////////////////////////////
// CZIPFile::File seek to a file
#pragma pack(1)
typedef struct
{
DWORD nSignature; // 0x04034b50
WORD nVersion;
WORD nFlags;
WORD nCompression;
WORD nFileTime;
WORD nFileDate;
DWORD nCRC;
DWORD nCompressedSize;
DWORD nActualSize;
WORD nNameLen;
WORD nExtraLen;
} ZIP_LOCAL_FILE;
#pragma pack()
BOOL CZIPFile::SeekToFile(File* pFile)
{
ASSERT( this != NULL );
ASSERT( pFile != NULL );
ASSERT( pFile->m_pZIP == this );
if ( m_hFile == INVALID_HANDLE_VALUE ) return FALSE;
if ( SetFilePointer( m_hFile, (DWORD)pFile->m_nLocalOffset, NULL, FILE_BEGIN )
!= pFile->m_nLocalOffset ) return FALSE;
ZIP_LOCAL_FILE pLocal;
DWORD nRead = 0;
ReadFile( m_hFile, &pLocal, sizeof(pLocal), &nRead, NULL );
if ( nRead != sizeof(pLocal) ) return FALSE;
if ( pLocal.nSignature != 0x04034b50 ) return FALSE;
if ( pLocal.nCompression != Z_DEFLATED && pLocal.nCompression != 0 ) return FALSE;
SetFilePointer( m_hFile, pLocal.nNameLen + pLocal.nExtraLen, NULL, FILE_CURRENT );
return TRUE;
}
/////////////////////////////////////////////////////////////////////////////
// CZIPFile::File prepare to decompress
BOOL CZIPFile::File::PrepareToDecompress(LPVOID pStream)
{
ZeroMemory( pStream, sizeof(z_stream) );
if ( ! m_pZIP->SeekToFile( this ) ) return FALSE;
if ( m_nCompression == 0 )
{
return ( m_nSize == m_nCompressedSize );
}
else
{
ASSERT( m_nCompression == Z_DEFLATED );
return Z_OK == inflateInit2( (z_stream*)pStream, -MAX_WBITS );
}
}
/////////////////////////////////////////////////////////////////////////////
// CZIPFile::File decompress to memory
/*CBuffer* CZIPFile::File::Decompress()
{
z_stream pStream;
if ( m_nSize > 32*1024*1024 ) return NULL;
if ( ! PrepareToDecompress( &pStream ) ) return NULL;
if ( m_nCompression == 0 )
{
CBuffer* pTarget = new CBuffer();
pTarget->EnsureBuffer( (DWORD)m_nSize );
ReadFile( m_pZIP->m_hFile, pTarget->m_pBuffer, (DWORD)m_nSize, &pTarget->m_nLength, NULL );
if ( pTarget->m_nLength == (DWORD)m_nSize ) return pTarget;
delete pTarget;
return NULL;
}
DWORD nSource = (DWORD)m_nCompressedSize;
BYTE* pSource = new BYTE[ nSource ];
ReadFile( m_pZIP->m_hFile, pSource, nSource, &nSource, NULL );
if ( nSource != (DWORD)m_nCompressedSize )
{
inflateEnd( &pStream );
return NULL;
}
CBuffer* pTarget = new CBuffer();
pTarget->EnsureBuffer( (DWORD)m_nSize );
pTarget->m_nLength = (DWORD)m_nSize;
pStream.next_in = pSource;
pStream.avail_in = (DWORD)m_nCompressedSize;
pStream.next_out = pTarget->m_pBuffer;
pStream.avail_out = pTarget->m_nLength;
inflate( &pStream, Z_FINISH );
delete [] pSource;
if ( pStream.avail_out != 0 )
{
delete pTarget;
pTarget = NULL;
}
inflateEnd( &pStream );
return pTarget;
}*/
/////////////////////////////////////////////////////////////////////////////
// CZIPFile::File decompress to disk
#define BUFFER_IN_SIZE (64*1024)
#define BUFFER_OUT_SIZE (128*1024)
BOOL CZIPFile::File::Extract(LPCTSTR pszFile)
{
z_stream pStream;
HANDLE hFile;
hFile = CreateFile( pszFile, GENERIC_WRITE, 0, NULL, CREATE_NEW,
FILE_ATTRIBUTE_NORMAL, NULL );
if ( hFile == INVALID_HANDLE_VALUE ) return FALSE;
if ( ! PrepareToDecompress( &pStream ) ) return NULL;
uint64 nCompressed = 0, nUncompressed = 0;
if ( m_nCompression == Z_DEFLATED )
{
BYTE* pBufferIn = new BYTE[BUFFER_IN_SIZE];
BYTE* pBufferOut = new BYTE[BUFFER_OUT_SIZE];
while ( nCompressed < m_nCompressedSize || nUncompressed < m_nSize )
{
if ( pStream.avail_in == 0 )
{
pStream.avail_in = (DWORD)min( m_nCompressedSize - nCompressed, BUFFER_IN_SIZE );
pStream.next_in = pBufferIn;
DWORD nRead = 0;
ReadFile( m_pZIP->m_hFile, pBufferIn, pStream.avail_in, &nRead, NULL );
if ( nRead != pStream.avail_in ) break;
nCompressed += nRead;
}
pStream.avail_out = BUFFER_OUT_SIZE;
pStream.next_out = pBufferOut;
int nInflate = inflate( &pStream, Z_SYNC_FLUSH );
if ( pStream.avail_out < BUFFER_OUT_SIZE )
{
DWORD nWrite = BUFFER_OUT_SIZE - pStream.avail_out;
WriteFile( hFile, pBufferOut, nWrite, &nWrite, NULL );
if ( nWrite != BUFFER_OUT_SIZE - pStream.avail_out ) break;
nUncompressed += nWrite;
}
}
delete [] pBufferOut;
delete [] pBufferIn;
inflateEnd( &pStream );
}
else
{
BYTE* pBufferOut = new BYTE[BUFFER_OUT_SIZE];
while ( nUncompressed < m_nSize )
{
DWORD nChunk = (DWORD)min( m_nSize - nUncompressed, BUFFER_OUT_SIZE );
DWORD nProcess = 0;
ReadFile( m_pZIP->m_hFile, pBufferOut, nChunk, &nProcess, NULL );
if ( nChunk != nProcess ) break;
WriteFile( hFile, pBufferOut, nChunk, &nProcess, NULL );
if ( nChunk != nProcess ) break;
nCompressed += nChunk;
nUncompressed += nChunk;
}
delete [] pBufferOut;
}
CloseHandle( hFile );
if ( nUncompressed >= m_nSize ) return TRUE;
DeleteFile( pszFile );
return FALSE;
}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -