static int CompareIndexEntries_FilePos(const void *, const void * pvIndexEntry1, const void * pvIndexEntry2) { PCASC_INDEX_ENTRY pIndexEntry1 = (PCASC_INDEX_ENTRY)pvIndexEntry1; PCASC_INDEX_ENTRY pIndexEntry2 = (PCASC_INDEX_ENTRY)pvIndexEntry2; ULONGLONG FileOffset1 = ConvertBytesToInteger_5(pIndexEntry1->FileOffsetBE); ULONGLONG FileOffset2 = ConvertBytesToInteger_5(pIndexEntry2->FileOffsetBE); DWORD ArchIndex1 = (DWORD)(FileOffset1 >> 0x1E); DWORD ArchIndex2 = (DWORD)(FileOffset2 >> 0x1E); // First, compare the archive index if(ArchIndex1 < ArchIndex2) return -1; if(ArchIndex1 > ArchIndex2) return +1; // Second, compare the archive offset FileOffset1 &= 0x3FFFFFFF; FileOffset2 &= 0x3FFFFFFF; if(FileOffset1 < FileOffset2) return -1; if(FileOffset1 > FileOffset2) return +1; return 0; }
static TCascFile * CreateFileHandle(TCascStorage * hs, PCASC_INDEX_ENTRY pIndexEntry) { ULONGLONG FileOffsMask = ((ULONGLONG)1 << hs->KeyMapping[0].SegmentBits) - 1; ULONGLONG FileOffset = ConvertBytesToInteger_5(pIndexEntry->FileOffsetBE); TCascFile * hf; // Allocate the CASC file structure hf = (TCascFile *)CASC_ALLOC(TCascFile, 1); if(hf != NULL) { // Initialize the structure memset(hf, 0, sizeof(TCascFile)); hf->ArchiveIndex = (DWORD)(FileOffset >> hs->KeyMapping[0].SegmentBits); hf->HeaderOffset = (DWORD)(FileOffset & FileOffsMask); hf->szClassName = "TCascFile"; // Copy the file size. Note that for all files except ENCODING, // this is the compressed file size hf->CompressedSize = ConvertBytesToInteger_4_LE(pIndexEntry->FileSizeLE); // For now, we set the file size to be equal to compressed size // This is used when loading the ENCODING file, which does not // have entry in the encoding table hf->FileSize = hf->CompressedSize; // Increment the number of references to the archive hs->dwRefCount++; hf->hs = hs; }
static void DumpIndexKey( FILE * fp, TCascStorage * hs, LPBYTE pbIndexKey, int nDumpLevel) { PCASC_INDEX_ENTRY pIndexEntry; TCascFile * hf; QUERY_KEY QueryKey; HANDLE hFile; BYTE HeaderArea[MAX_HEADER_AREA_SIZE]; char szBuffer[0x20]; QueryKey.pbData = pbIndexKey; QueryKey.cbData = MD5_HASH_SIZE; pIndexEntry = FindIndexEntry(hs, &QueryKey); if(pIndexEntry != NULL) { ULONGLONG FileOffset = ConvertBytesToInteger_5(pIndexEntry->FileOffsetBE); DWORD ArchIndex = (DWORD)(FileOffset >> 0x1E); DWORD FileSize = ConvertBytesToInteger_4_LE(pIndexEntry->FileSizeLE); // Mask the file offset FileOffset &= 0x3FFFFFFF; fprintf(fp, " data.%03u at 0x%08x (0x%lx bytes)\n", ArchIndex, (DWORD)FileOffset, FileSize); if(nDumpLevel > 2) { QueryKey.pbData = pIndexEntry->IndexKey; QueryKey.cbData = MD5_HASH_SIZE; if(CascOpenFileByIndexKey((HANDLE)hs, &QueryKey, 0, &hFile)) { // Make sure that the data file is open and frame header loaded CascGetFileSize(hFile, NULL); hf = IsValidFileHandle(hFile); assert(hf->pStream != NULL); // Read the header area FileOffset = hf->HeaderOffset - BLTE_HEADER_DELTA; FileStream_Read(hf->pStream, &FileOffset, HeaderArea, sizeof(HeaderArea)); CascCloseFile(hFile); // Dump the header area fprintf(fp, " FileSize: %X Rest: %s\n", ConvertBytesToInteger_4_LE(&HeaderArea[0x10]), StringFromBinary(&HeaderArea[0x14], 10, szBuffer)); } } }
static TCascFile * CreateFileHandle(TCascStorage * hs, PQUERY_KEY pCKey, PQUERY_KEY pEKey, PCASC_EKEY_ENTRY pEKeyEntry, DWORD dwOpenFlags, DWORD dwContentSize) { ULONGLONG StorageOffset = ConvertBytesToInteger_5(pEKeyEntry->StorageOffset); ULONGLONG FileOffsMask = ((ULONGLONG)1 << hs->IndexFile[0].FileOffsetBits) - 1; TCascFile * hf; // Allocate the CASC file structure hf = (TCascFile *)CASC_ALLOC(TCascFile, 1); if(hf != NULL) { // Initialize the structure memset(hf, 0, sizeof(TCascFile)); hf->ArchiveIndex = (DWORD)(StorageOffset >> hs->IndexFile[0].FileOffsetBits); hf->ArchiveOffset = (DWORD)(StorageOffset & FileOffsMask); hf->szClassName = "TCascFile"; hf->OpenFlags = dwOpenFlags; // Supply the content key, if available if(pCKey != NULL) { assert(pCKey->pbData != NULL && pCKey->cbData == MD5_HASH_SIZE); memcpy(hf->CKey.Value, pCKey->pbData, pCKey->cbData); } // Supply the encoded key, if available if(pEKey != NULL) { assert(pEKey->pbData != NULL && CASC_EKEY_SIZE <= pEKey->cbData && pEKey->cbData <= MD5_HASH_SIZE); memcpy(hf->EKey.Value, pEKey->pbData, pEKey->cbData); } // Copy the encoded file size if(pEKeyEntry != NULL) { hf->EncodedSize = ConvertBytesToInteger_4_LE(pEKeyEntry->EncodedSize); } // Set the content size hf->ContentSize = dwContentSize; // Increment the number of references to the archive hs->dwRefCount++; hf->hs = hs; }
void CascDumpIndexEntry( TCascStorage * /* hs */, TDumpContext * dc, PCASC_INDEX_ENTRY pIndexEntry, int /* nDumpLevel */) { if(pIndexEntry != NULL) { ULONGLONG FileOffset = ConvertBytesToInteger_5(pIndexEntry->FileOffsetBE); DWORD ArchIndex = (DWORD)(FileOffset >> 0x1E); DWORD FileSize = ConvertBytesToInteger_4_LE(pIndexEntry->FileSizeLE); // Mask the file offset FileOffset &= 0x3FFFFFFF; dump_print(dc, " data.%03u at 0x%08x (0x%lx bytes)\n", ArchIndex, (DWORD)FileOffset, FileSize); //if(nDumpLevel > 2) //{ // QueryKey.pbData = pIndexEntry->IndexKey; // QueryKey.cbData = MD5_HASH_SIZE; // if(CascOpenFileByIndexKey((HANDLE)hs, &QueryKey, 0, &hFile)) // { // // Make sure that the data file is open and frame header loaded // CascGetFileSize(hFile, NULL); // hf = IsValidFileHandle(hFile); // assert(hf->pStream != NULL); // // Read the header area // FileOffset = hf->HeaderOffset - BLTE_HEADER_DELTA; // FileStream_Read(hf->pStream, &FileOffset, HeaderArea, sizeof(HeaderArea)); // CascCloseFile(hFile); // // Dump the header area // dump_print(dc, " FileSize: %X Rest: %s\n", // ConvertBytesToInteger_4_LE(&HeaderArea[0x10]), // StringFromBinary(&HeaderArea[0x14], 10, szBuffer)); // } //} }