2014-02-13 05:47:20 -05:00
// MCADefrag.cpp
// Implements the main app entrypoint and the cMCADefrag class representing the entire app
# include "Globals.h"
# include "MCADefrag.h"
2014-08-12 11:05:04 -04:00
# include "Logger.h"
# include "LoggerListeners.h"
2014-02-13 10:54:29 -05:00
# include "zlib/zlib.h"
2014-02-13 05:47:20 -05:00
2014-02-13 06:48:22 -05:00
// An array of 4096 zero bytes, used for writing the padding
static const Byte g_Zeroes [ 4096 ] = { 0 } ;
2014-02-13 05:47:20 -05:00
int main ( int argc , char * * argv )
{
2015-08-30 17:57:43 -04:00
auto consoleLogListener = MakeConsoleListener ( false ) ;
auto consoleAttachment = cLogger : : GetInstance ( ) . AttachListener ( std : : move ( consoleLogListener ) ) ;
auto fileLogListenerRet = MakeFileListener ( ) ;
if ( ! fileLogListenerRet . first )
{
LOGERROR ( " Failed to open log file, aborting " ) ;
return EXIT_FAILURE ;
}
auto fileAttachment = cLogger : : GetInstance ( ) . AttachListener ( std : : move ( fileLogListenerRet . second ) ) ;
2014-08-10 15:10:47 -04:00
2014-08-12 11:05:04 -04:00
cLogger : : InitiateMultithreading ( ) ;
2014-08-10 15:10:47 -04:00
2014-02-13 05:47:20 -05:00
cMCADefrag Defrag ;
if ( ! Defrag . Init ( argc , argv ) )
{
2015-08-30 17:57:43 -04:00
return EXIT_FAILURE ;
2014-02-13 05:47:20 -05:00
}
Defrag . Run ( ) ;
return 0 ;
}
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// cMCADefrag:
cMCADefrag : : cMCADefrag ( void ) :
2014-02-13 10:54:29 -05:00
m_NumThreads ( 4 ) ,
m_ShouldRecompress ( true )
2014-02-13 05:47:20 -05:00
{
}
bool cMCADefrag : : Init ( int argc , char * * argv )
{
// Nothing needed yet
return true ;
}
void cMCADefrag : : Run ( void )
{
// Fill the queue with MCA files
m_Queue = cFile : : GetFolderContents ( " . " ) ;
// Start the processing threads:
for ( int i = 0 ; i < m_NumThreads ; i + + )
{
StartThread ( ) ;
}
// Wait for all the threads to finish:
while ( ! m_Threads . empty ( ) )
{
m_Threads . front ( ) - > Wait ( ) ;
delete m_Threads . front ( ) ;
m_Threads . pop_front ( ) ;
}
}
void cMCADefrag : : StartThread ( void )
{
cThread * Thread = new cThread ( * this ) ;
m_Threads . push_back ( Thread ) ;
Thread - > Start ( ) ;
}
AString cMCADefrag : : GetNextFileName ( void )
{
cCSLock Lock ( m_CS ) ;
if ( m_Queue . empty ( ) )
{
return AString ( ) ;
}
AString res = m_Queue . back ( ) ;
m_Queue . pop_back ( ) ;
return res ;
}
///////////////////////////////////////////////////////////////////////////////////////////////////////////////////////
// cMCADefrag::cThread:
cMCADefrag : : cThread : : cThread ( cMCADefrag & a_Parent ) :
super ( " MCADefrag thread " ) ,
2014-02-13 10:54:29 -05:00
m_Parent ( a_Parent ) ,
m_IsChunkUncompressed ( false )
2014-02-13 05:47:20 -05:00
{
}
void cMCADefrag : : cThread : : Execute ( void )
{
for ( ; ; )
{
AString FileName = m_Parent . GetNextFileName ( ) ;
if ( FileName . empty ( ) )
{
return ;
}
ProcessFile ( FileName ) ;
}
}
void cMCADefrag : : cThread : : ProcessFile ( const AString & a_FileName )
{
// Filter out non-MCA files:
if ( ( a_FileName . length ( ) < 4 ) | | ( a_FileName . substr ( a_FileName . length ( ) - 4 , 4 ) ! = " .mca " ) )
{
return ;
}
LOGINFO ( " %s " , a_FileName . c_str ( ) ) ;
// Open input and output files:
AString OutFileName = a_FileName + " .new " ;
cFile In , Out ;
if ( ! In . Open ( a_FileName , cFile : : fmRead ) )
{
LOGWARNING ( " Cannot open file %s for reading, skipping file. " , a_FileName . c_str ( ) ) ;
return ;
}
if ( ! Out . Open ( OutFileName . c_str ( ) , cFile : : fmWrite ) )
{
LOGWARNING ( " Cannot open file %s for writing, skipping file. " , OutFileName . c_str ( ) ) ;
return ;
}
// Read the Locations and Timestamps from the input file:
Byte Locations [ 4096 ] ;
UInt32 Timestamps [ 1024 ] ;
if ( In . Read ( Locations , sizeof ( Locations ) ) ! = sizeof ( Locations ) )
{
LOGWARNING ( " Cannot read Locations in file %s, skipping file. " , a_FileName . c_str ( ) ) ;
return ;
}
if ( In . Read ( Timestamps , sizeof ( Timestamps ) ) ! = sizeof ( Timestamps ) )
{
LOGWARNING ( " Cannot read Timestamps in file %s, skipping file. " , a_FileName . c_str ( ) ) ;
return ;
}
// Write dummy Locations to the Out file (will be overwritten once the correct ones are known)
if ( Out . Write ( Locations , sizeof ( Locations ) ) ! = sizeof ( Locations ) )
{
LOGWARNING ( " Cannot write Locations to file %s, skipping file. " , OutFileName . c_str ( ) ) ;
return ;
}
m_CurrentSectorOut = 2 ;
// Write a copy of the Timestamps into the Out file:
if ( Out . Write ( Timestamps , sizeof ( Timestamps ) ) ! = sizeof ( Timestamps ) )
{
LOGWARNING ( " Cannot write Timestamps to file %s, skipping file. " , OutFileName . c_str ( ) ) ;
return ;
}
// Process each chunk:
for ( size_t i = 0 ; i < 1024 ; i + + )
{
size_t idx = i * 4 ;
if (
( Locations [ idx ] = = 0 ) & &
( Locations [ idx + 1 ] = = 0 ) & &
( Locations [ idx + 2 ] = = 0 ) & &
( Locations [ idx + 3 ] = = 0 )
)
{
// Chunk not present
continue ;
}
2014-02-13 10:54:29 -05:00
m_IsChunkUncompressed = false ;
2014-02-13 05:47:20 -05:00
if ( ! ReadChunk ( In , Locations + idx ) )
{
LOGWARNING ( " Cannot read chunk #%d from file %s. Skipping file. " , i , a_FileName . c_str ( ) ) ;
return ;
}
if ( ! WriteChunk ( Out , Locations + idx ) )
{
LOGWARNING ( " Cannot write chunk #%d to file %s. Skipping file. " , i , OutFileName . c_str ( ) ) ;
return ;
}
}
// Write the new Locations into the MCA header:
Out . Seek ( 0 ) ;
if ( Out . Write ( Locations , sizeof ( Locations ) ) ! = sizeof ( Locations ) )
{
LOGWARNING ( " Cannot write updated Locations to file %s, skipping file. " , OutFileName . c_str ( ) ) ;
return ;
}
// Close the files, delete orig, rename new:
In . Close ( ) ;
Out . Close ( ) ;
cFile : : Delete ( a_FileName ) ;
cFile : : Rename ( OutFileName , a_FileName ) ;
}
bool cMCADefrag : : cThread : : ReadChunk ( cFile & a_File , const Byte * a_LocationRaw )
{
int SectorNum = ( a_LocationRaw [ 0 ] < < 16 ) | ( a_LocationRaw [ 1 ] < < 8 ) | a_LocationRaw [ 2 ] ;
int SizeInSectors = a_LocationRaw [ 3 ] * ( 4 KiB ) ;
if ( a_File . Seek ( SectorNum * ( 4 KiB ) ) < 0 )
{
2016-08-24 15:45:03 -04:00
LOGWARNING ( " Failed to seek to chunk data - file pos %llu (%d KiB, %.02f MiB)! " ,
static_cast < Int64 > ( SectorNum ) * ( 4 KiB ) , SectorNum * 4 ,
static_cast < double > ( SectorNum ) / 256
) ;
2014-02-13 05:47:20 -05:00
return false ;
}
// Read the exact size:
Byte Buf [ 4 ] ;
if ( a_File . Read ( Buf , 4 ) ! = 4 )
{
LOGWARNING ( " Failed to read chunk data length " ) ;
return false ;
}
m_CompressedChunkDataSize = ( Buf [ 0 ] < < 24 ) | ( Buf [ 1 ] < < 16 ) | ( Buf [ 2 ] < < 8 ) | Buf [ 3 ] ;
2014-12-21 16:50:32 -05:00
if ( ( m_CompressedChunkDataSize > SizeInSectors ) | | ( m_CompressedChunkDataSize < 0 ) )
2014-02-13 05:47:20 -05:00
{
LOGWARNING ( " Invalid chunk data - SizeInSectors (%d) smaller that RealSize (%d) " , SizeInSectors , m_CompressedChunkDataSize ) ;
return false ;
}
// Read the data:
2016-08-24 15:45:03 -04:00
if ( a_File . Read ( m_CompressedChunkData , static_cast < size_t > ( m_CompressedChunkDataSize ) ) ! = m_CompressedChunkDataSize )
2014-02-13 05:47:20 -05:00
{
LOGWARNING ( " Failed to read chunk data! " ) ;
return false ;
}
2014-02-13 10:54:29 -05:00
// Uncompress the data if recompression is active
if ( m_Parent . m_ShouldRecompress )
{
m_IsChunkUncompressed = UncompressChunk ( ) ;
if ( ! m_IsChunkUncompressed )
{
LOGINFO ( " Chunk failed to uncompress, will be copied verbatim instead. " ) ;
}
}
2014-02-13 05:47:20 -05:00
return true ;
}
bool cMCADefrag : : cThread : : WriteChunk ( cFile & a_File , Byte * a_LocationRaw )
{
2014-02-13 10:54:29 -05:00
// Recompress the data if recompression is active:
if ( m_Parent . m_ShouldRecompress )
{
if ( ! CompressChunk ( ) )
{
LOGINFO ( " Chunk failed to recompress, will be coped verbatim instead. " ) ;
}
}
2014-02-13 05:47:20 -05:00
2014-02-13 10:54:29 -05:00
// Update the Location:
2016-08-24 15:45:03 -04:00
a_LocationRaw [ 0 ] = static_cast < Byte > ( m_CurrentSectorOut > > 16 ) ;
2014-02-13 05:47:20 -05:00
a_LocationRaw [ 1 ] = ( m_CurrentSectorOut > > 8 ) & 0xff ;
a_LocationRaw [ 2 ] = m_CurrentSectorOut & 0xff ;
2016-08-24 15:45:03 -04:00
a_LocationRaw [ 3 ] = static_cast < Byte > ( ( m_CompressedChunkDataSize + ( 4 KiB ) + 3 ) / ( 4 KiB ) ) ; // +3 because the m_CompressedChunkDataSize doesn't include the exact-length
2014-02-13 10:54:29 -05:00
m_CurrentSectorOut + = a_LocationRaw [ 3 ] ;
2014-02-13 05:47:20 -05:00
// Write the data length:
Byte Buf [ 4 ] ;
2016-08-24 15:45:03 -04:00
Buf [ 0 ] = static_cast < Byte > ( m_CompressedChunkDataSize > > 24 ) ;
2014-02-13 05:47:20 -05:00
Buf [ 1 ] = ( m_CompressedChunkDataSize > > 16 ) & 0xff ;
Buf [ 2 ] = ( m_CompressedChunkDataSize > > 8 ) & 0xff ;
Buf [ 3 ] = m_CompressedChunkDataSize & 0xff ;
if ( a_File . Write ( Buf , 4 ) ! = 4 )
{
LOGWARNING ( " Failed to write chunk length! " ) ;
return false ;
}
// Write the data:
2016-08-24 15:45:03 -04:00
if ( a_File . Write ( m_CompressedChunkData , static_cast < size_t > ( m_CompressedChunkDataSize ) ) ! = m_CompressedChunkDataSize )
2014-02-13 05:47:20 -05:00
{
LOGWARNING ( " Failed to write chunk data! " ) ;
return false ;
}
2014-02-13 06:48:22 -05:00
// Pad onto the next sector:
int NumPadding = a_LocationRaw [ 3 ] * 4096 - ( m_CompressedChunkDataSize + 4 ) ;
ASSERT ( NumPadding > = 0 ) ;
2016-08-24 15:45:03 -04:00
if ( ( NumPadding > 0 ) & & ( a_File . Write ( g_Zeroes , static_cast < size_t > ( NumPadding ) ) ! = NumPadding ) )
2014-02-13 06:48:22 -05:00
{
LOGWARNING ( " Failed to write padding " ) ;
return false ;
}
2014-02-13 10:54:29 -05:00
return true ;
}
bool cMCADefrag : : cThread : : UncompressChunk ( void )
{
switch ( m_CompressedChunkData [ 0 ] )
{
case COMPRESSION_GZIP : return UncompressChunkGzip ( ) ;
case COMPRESSION_ZLIB : return UncompressChunkZlib ( ) ;
}
LOGINFO ( " Chunk is compressed with in an unknown algorithm " ) ;
return false ;
}
bool cMCADefrag : : cThread : : UncompressChunkGzip ( void )
{
// TODO
// This format is not used in practice
return false ;
}
bool cMCADefrag : : cThread : : UncompressChunkZlib ( void )
{
// Uncompress the data:
z_stream strm ;
2016-08-24 15:45:03 -04:00
strm . zalloc = nullptr ;
strm . zfree = nullptr ;
strm . opaque = nullptr ;
2014-02-13 10:54:29 -05:00
inflateInit ( & strm ) ;
strm . next_out = m_RawChunkData ;
strm . avail_out = sizeof ( m_RawChunkData ) ;
strm . next_in = m_CompressedChunkData + 1 ; // The first byte is the compression method, skip it
2016-08-24 15:45:03 -04:00
strm . avail_in = static_cast < uInt > ( m_CompressedChunkDataSize ) ;
2014-02-13 10:54:29 -05:00
int res = inflate ( & strm , Z_FINISH ) ;
inflateEnd ( & strm ) ;
if ( res ! = Z_STREAM_END )
{
LOGWARNING ( " Failed to uncompress chunk data: %s " , strm . msg ) ;
return false ;
}
2016-08-24 15:45:03 -04:00
ASSERT ( strm . total_out < static_cast < uLong > ( std : : numeric_limits < int > : : max ( ) ) ) ;
m_RawChunkDataSize = static_cast < int > ( strm . total_out ) ;
2014-02-13 10:54:29 -05:00
return true ;
}
bool cMCADefrag : : cThread : : CompressChunk ( void )
{
// Check that the compressed data can fit:
2016-08-24 15:45:03 -04:00
uLongf CompressedSize = compressBound ( static_cast < uLong > ( m_RawChunkDataSize ) ) ;
2014-02-13 10:54:29 -05:00
if ( CompressedSize > sizeof ( m_CompressedChunkData ) )
{
LOGINFO ( " Too much data for the internal compression buffer! " ) ;
return false ;
}
// Compress the data using the highest compression factor:
2016-08-24 15:45:03 -04:00
int errorcode = compress2 ( m_CompressedChunkData + 1 , & CompressedSize , m_RawChunkData , static_cast < uLong > ( m_RawChunkDataSize ) , Z_BEST_COMPRESSION ) ;
2014-02-13 10:54:29 -05:00
if ( errorcode ! = Z_OK )
{
LOGINFO ( " Recompression failed: %d " , errorcode ) ;
return false ;
}
m_CompressedChunkData [ 0 ] = COMPRESSION_ZLIB ;
2016-08-24 15:45:03 -04:00
ASSERT ( CompressedSize < static_cast < uLong > ( std : : numeric_limits < int > : : max ( ) ) ) ;
m_CompressedChunkDataSize = static_cast < int > ( CompressedSize + 1 ) ;
2014-02-13 05:47:20 -05:00
return true ;
}