aboutsummaryrefslogtreecommitdiff
path: root/dep/src/libmpq/mpq.c
diff options
context:
space:
mode:
Diffstat (limited to 'dep/src/libmpq/mpq.c')
-rw-r--r--dep/src/libmpq/mpq.c1033
1 files changed, 0 insertions, 1033 deletions
diff --git a/dep/src/libmpq/mpq.c b/dep/src/libmpq/mpq.c
deleted file mode 100644
index 4e3d7db0b4d..00000000000
--- a/dep/src/libmpq/mpq.c
+++ /dev/null
@@ -1,1033 +0,0 @@
-/*
- * mpq.c -- functions for developers using libmpq.
- *
- * Copyright (c) 2003-2008 Maik Broemme <mbroemme@plusserver.de>
- *
- * This program 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.
- *
- * This program 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 this program; if not, write to the Free Software
- * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
- */
-
-/* mpq-tools configuration includes. */
-#include "config.h"
-
-/* libmpq main includes. */
-#include "mpq.h"
-#include "mpq-internal.h"
-
-/* libmpq generic includes. */
-#include "common.h"
-
-/* generic includes. */
-#include <fcntl.h>
-#include <stdlib.h>
-#include <string.h>
-#include <sys/stat.h>
-#if HAVE_UNISTD_H
- #include <unistd.h>
-#endif
-
-#ifdef _MSC_VER
- #define fseeko _fseeki64
-#endif
-
-/* this function returns the library version information. */
-const char *libmpq__version(void) {
-
- /* return version information. */
- return VERSION;
-}
-
-/* this function read a file and verify if it is a valid mpq archive, then it read and decrypt the hash table. */
-int32_t libmpq__archive_open(mpq_archive_s **mpq_archive, const char *mpq_filename, libmpq__off_t archive_offset) {
-
- /* some common variables. */
- uint32_t rb = 0;
- uint32_t i = 0;
- uint32_t count = 0;
- int32_t result = 0;
- uint32_t header_search = FALSE;
-
- if (archive_offset == -1) {
- archive_offset = 0;
- header_search = TRUE;
- }
-
- if ((*mpq_archive = calloc(1, sizeof(mpq_archive_s))) == NULL) {
-
- /* archive struct could not be allocated */
- return LIBMPQ_ERROR_MALLOC;
- }
-
- /* check if file exists and is readable */
- if (((*mpq_archive)->fp = fopen(mpq_filename, "rb")) == NULL) {
-
- /* file could not be opened. */
- result = LIBMPQ_ERROR_OPEN;
- goto error;
- }
-
- /* assign some default values. */
- (*mpq_archive)->mpq_header.mpq_magic = 0;
- (*mpq_archive)->files = 0;
-
- /* loop through file and search for mpq signature. */
- while (TRUE) {
-
- /* reset header values. */
- (*mpq_archive)->mpq_header.mpq_magic = 0;
-
- /* seek in file. */
- if (fseeko((*mpq_archive)->fp, archive_offset, SEEK_SET) < 0) {
-
- /* seek in file failed. */
- result = LIBMPQ_ERROR_SEEK;
- goto error;
- }
-
- /* read header from file. */
- if ((rb = fread(&(*mpq_archive)->mpq_header, 1, sizeof(mpq_header_s), (*mpq_archive)->fp)) != sizeof(mpq_header_s)) {
-
- /* no valid mpq archive. */
- result = LIBMPQ_ERROR_FORMAT;
- goto error;
- }
-
- /* check if we found a valid mpq header. */
- if ((*mpq_archive)->mpq_header.mpq_magic == LIBMPQ_HEADER) {
-
- /* check if we process old mpq archive version. */
- if ((*mpq_archive)->mpq_header.version == LIBMPQ_ARCHIVE_VERSION_ONE) {
-
- /* check if the archive is protected. */
- if ((*mpq_archive)->mpq_header.header_size != sizeof(mpq_header_s)) {
-
- /* correct header size. */
- (*mpq_archive)->mpq_header.header_size = sizeof(mpq_header_s);
- }
- }
-
- /* check if we process new mpq archive version. */
- if ((*mpq_archive)->mpq_header.version == LIBMPQ_ARCHIVE_VERSION_TWO) {
-
- /* check if the archive is protected. */
- if ((*mpq_archive)->mpq_header.header_size != sizeof(mpq_header_s) + sizeof(mpq_header_ex_s)) {
-
- /* correct header size. */
- (*mpq_archive)->mpq_header.header_size = sizeof(mpq_header_s) + sizeof(mpq_header_ex_s);
- }
- }
-
- /* break the loop, because header was found. */
- break;
- }
-
- /* move to the next possible offset. */
- if (!header_search) {
-
- /* no valid mpq archive. */
- result = LIBMPQ_ERROR_FORMAT;
- goto error;
- }
- archive_offset += 512;
- }
-
- /* store block size for later use. */
- (*mpq_archive)->block_size = 512 << (*mpq_archive)->mpq_header.block_size;
-
- /* store archive offset and size for later use. */
- (*mpq_archive)->archive_offset = archive_offset;
-
- /* check if we process new mpq archive version. */
- if ((*mpq_archive)->mpq_header.version == LIBMPQ_ARCHIVE_VERSION_TWO) {
-
- /* seek in file. */
- if (fseeko((*mpq_archive)->fp, sizeof(mpq_header_s) + archive_offset, SEEK_SET) < 0) {
-
- /* seek in file failed. */
- result = LIBMPQ_ERROR_SEEK;
- goto error;
- }
-
- /* read header from file. */
- if ((rb = fread(&(*mpq_archive)->mpq_header_ex, 1, sizeof(mpq_header_ex_s), (*mpq_archive)->fp)) != sizeof(mpq_header_ex_s)) {
-
- /* no valid mpq archive. */
- result = LIBMPQ_ERROR_FORMAT;
- goto error;
- }
- }
-
- /* allocate memory for the block table, hash table, file and block table to file mapping. */
- if (((*mpq_archive)->mpq_block = calloc((*mpq_archive)->mpq_header.block_table_count, sizeof(mpq_block_s))) == NULL ||
- ((*mpq_archive)->mpq_block_ex = calloc((*mpq_archive)->mpq_header.block_table_count, sizeof(mpq_block_ex_s))) == NULL ||
- ((*mpq_archive)->mpq_hash = calloc((*mpq_archive)->mpq_header.hash_table_count, sizeof(mpq_hash_s))) == NULL ||
- ((*mpq_archive)->mpq_file = calloc((*mpq_archive)->mpq_header.block_table_count, sizeof(mpq_file_s))) == NULL ||
- ((*mpq_archive)->mpq_map = calloc((*mpq_archive)->mpq_header.block_table_count, sizeof(mpq_map_s))) == NULL) {
-
- /* memory allocation problem. */
- result = LIBMPQ_ERROR_MALLOC;
- goto error;
- }
-
- /* seek in file. */
- if (fseeko((*mpq_archive)->fp, (*mpq_archive)->mpq_header.hash_table_offset + (((long long)((*mpq_archive)->mpq_header_ex.hash_table_offset_high)) << 32) + (*mpq_archive)->archive_offset, SEEK_SET) < 0) {
-
- /* seek in file failed. */
- result = LIBMPQ_ERROR_SEEK;
- goto error;
- }
-
- /* read the hash table into the buffer. */
- if ((rb = fread((*mpq_archive)->mpq_hash, 1, (*mpq_archive)->mpq_header.hash_table_count * sizeof(mpq_hash_s), (*mpq_archive)->fp)) < 0) {
-
- /* something on read failed. */
- result = LIBMPQ_ERROR_READ;
- goto error;
- }
-
- /* decrypt the hashtable. */
- libmpq__decrypt_block((uint32_t *)((*mpq_archive)->mpq_hash), (*mpq_archive)->mpq_header.hash_table_count * sizeof(mpq_hash_s), libmpq__hash_string("(hash table)", 0x300));
-
- /* seek in file. */
- if (fseeko((*mpq_archive)->fp, (*mpq_archive)->mpq_header.block_table_offset + (((long long)((*mpq_archive)->mpq_header_ex.block_table_offset_high)) << 32) + (*mpq_archive)->archive_offset, SEEK_SET) < 0) {
-
- /* seek in file failed. */
- result = LIBMPQ_ERROR_SEEK;
- goto error;
- }
-
- /* read the block table into the buffer. */
- if ((rb = fread((*mpq_archive)->mpq_block, 1, (*mpq_archive)->mpq_header.block_table_count * sizeof(mpq_block_s), (*mpq_archive)->fp)) < 0) {
-
- /* something on read failed. */
- result = LIBMPQ_ERROR_READ;
- goto error;
- }
-
- /* decrypt block table. */
- libmpq__decrypt_block((uint32_t *)((*mpq_archive)->mpq_block), (*mpq_archive)->mpq_header.block_table_count * sizeof(mpq_block_s), libmpq__hash_string("(block table)", 0x300));
-
- /* check if extended block table is present, regardless of version 2 it is only present in archives > 4GB. */
- if ((*mpq_archive)->mpq_header_ex.extended_offset > 0) {
-
- /* seek in file. */
- if (fseeko((*mpq_archive)->fp, (*mpq_archive)->mpq_header_ex.extended_offset + archive_offset, SEEK_SET) < 0) {
-
- /* seek in file failed. */
- result = LIBMPQ_ERROR_SEEK;
- goto error;
- }
-
- /* read header from file. */
- if ((rb = fread((*mpq_archive)->mpq_block_ex, 1, (*mpq_archive)->mpq_header.block_table_count * sizeof(mpq_block_ex_s), (*mpq_archive)->fp)) < 0) {
-
- /* no valid mpq archive. */
- result = LIBMPQ_ERROR_FORMAT;
- goto error;
- }
- }
-
- /* loop through all files in mpq archive and check if they are valid. */
- for (i = 0; i < (*mpq_archive)->mpq_header.block_table_count; i++) {
-
- /* save block difference between valid and invalid blocks. */
- (*mpq_archive)->mpq_map[i].block_table_diff = i - count;
-
- /* check if file exists, sizes and offsets are correct. */
- if (((*mpq_archive)->mpq_block[i].flags & LIBMPQ_FLAG_EXISTS) == 0) {
-
- /* file does not exist, so nothing to do with that block. */
- continue;
- }
-
- /* create final indices tables. */
- (*mpq_archive)->mpq_map[count].block_table_indices = i;
-
- /* increase file counter. */
- count++;
- }
-
- /* save the number of files. */
- (*mpq_archive)->files = count;
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-
-error:
- if ((*mpq_archive)->fp)
- fclose((*mpq_archive)->fp);
-
- free((*mpq_archive)->mpq_map);
- free((*mpq_archive)->mpq_file);
- free((*mpq_archive)->mpq_hash);
- free((*mpq_archive)->mpq_block);
- free((*mpq_archive)->mpq_block_ex);
- free(*mpq_archive);
-
- *mpq_archive = NULL;
-
- return result;
-}
-
-/* this function close the file descriptor, free the decryption buffer and the file list. */
-int32_t libmpq__archive_close(mpq_archive_s *mpq_archive) {
-
- /* try to close the file */
- if ((fclose(mpq_archive->fp)) < 0) {
-
- /* don't free anything here, so the caller can try calling us
- * again.
- */
- return LIBMPQ_ERROR_CLOSE;
- }
-
- /* free header, tables and list. */
- free(mpq_archive->mpq_map);
- free(mpq_archive->mpq_file);
- free(mpq_archive->mpq_hash);
- free(mpq_archive->mpq_block);
- free(mpq_archive->mpq_block_ex);
- free(mpq_archive);
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-}
-
-/* this function return the packed size of all files in the archive. */
-int32_t libmpq__archive_packed_size(mpq_archive_s *mpq_archive, libmpq__off_t *packed_size) {
-
- /* some common variables. */
- uint32_t i;
-
- /* loop through all files in archive and count packed size. */
- for (i = 0; i < mpq_archive->files; i++) {
- *packed_size += mpq_archive->mpq_block[mpq_archive->mpq_map[i].block_table_indices].packed_size;
- }
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-}
-
-/* this function return the unpacked size of all files in the archive. */
-int32_t libmpq__archive_unpacked_size(mpq_archive_s *mpq_archive, libmpq__off_t *unpacked_size) {
-
- /* some common variables. */
- uint32_t i;
-
- /* loop through all files in archive and count unpacked size. */
- for (i = 0; i < mpq_archive->files; i++) {
- *unpacked_size += mpq_archive->mpq_block[mpq_archive->mpq_map[i].block_table_indices].unpacked_size;
- }
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-}
-
-/* this function return the archive offset (beginning of archive in file). */
-int32_t libmpq__archive_offset(mpq_archive_s *mpq_archive, libmpq__off_t *offset) {
-
- /* return archive offset. */
- *offset = mpq_archive->archive_offset;
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-}
-
-/* this function return the archive offset. */
-int32_t libmpq__archive_version(mpq_archive_s *mpq_archive, uint32_t *version) {
-
- /* return archive version. */
- *version = mpq_archive->mpq_header.version + 1;
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-}
-
-/* this function return the number of valid files in archive. */
-int32_t libmpq__archive_files(mpq_archive_s *mpq_archive, uint32_t *files) {
-
- /* return archive version. */
- *files = mpq_archive->files;
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-}
-
-/* this function return the packed size of the given files in the archive. */
-int32_t libmpq__file_packed_size(mpq_archive_s *mpq_archive, uint32_t file_number, libmpq__off_t *packed_size) {
-
- /* check if given file number is not out of range. */
- if (file_number < 0 || file_number > mpq_archive->files - 1) {
-
- /* file number is out of range. */
- return LIBMPQ_ERROR_EXIST;
- }
-
- /* get the packed size of file. */
- *packed_size = mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].packed_size;
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-}
-
-/* this function return the unpacked size of the given file in the archive. */
-int32_t libmpq__file_unpacked_size(mpq_archive_s *mpq_archive, uint32_t file_number, libmpq__off_t *unpacked_size) {
-
- /* check if given file number is not out of range. */
- if (file_number < 0 || file_number > mpq_archive->files - 1) {
-
- /* file number is out of range. */
- return LIBMPQ_ERROR_EXIST;
- }
-
- /* get the unpacked size of file. */
- *unpacked_size = mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].unpacked_size;
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-}
-
-/* this function return the file offset (beginning of file in archive). */
-int32_t libmpq__file_offset(mpq_archive_s *mpq_archive, uint32_t file_number, libmpq__off_t *offset) {
-
- /* check if given file number is not out of range. */
- if (file_number < 0 || file_number > mpq_archive->files - 1) {
-
- /* file number is out of range. */
- return LIBMPQ_ERROR_EXIST;
- }
-
- /* return file offset relative to archive start. */
- *offset = mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].offset + (((long long)mpq_archive->mpq_block_ex[mpq_archive->mpq_map[file_number].block_table_indices].offset_high) << 32);
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-}
-
-/* this function return the number of blocks for the given file in the archive. */
-int32_t libmpq__file_blocks(mpq_archive_s *mpq_archive, uint32_t file_number, uint32_t *blocks) {
-
- /* check if given file number is not out of range. */
- if (file_number < 0 || file_number > mpq_archive->files - 1) {
-
- /* file number is out of range. */
- return LIBMPQ_ERROR_EXIST;
- }
-
- /* return the number of blocks for the given file. */
- *blocks = (mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].flags & LIBMPQ_FLAG_SINGLE) != 0 ? 1 : (mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].unpacked_size + mpq_archive->block_size - 1) / mpq_archive->block_size;
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-}
-
-/* this function return if the file is encrypted or not. */
-int32_t libmpq__file_encrypted(mpq_archive_s *mpq_archive, uint32_t file_number, uint32_t *encrypted) {
-
- /* check if given file number is not out of range. */
- if (file_number < 0 || file_number > mpq_archive->files - 1) {
-
- /* file number is out of range. */
- return LIBMPQ_ERROR_EXIST;
- }
-
- /* return the encryption status of file. */
- *encrypted = (mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].flags & LIBMPQ_FLAG_ENCRYPTED) != 0 ? TRUE : FALSE;
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-}
-
-/* this function return if the file is compressed or not. */
-int32_t libmpq__file_compressed(mpq_archive_s *mpq_archive, uint32_t file_number, uint32_t *compressed) {
-
- /* check if given file number is not out of range. */
- if (file_number < 0 || file_number > mpq_archive->files - 1) {
-
- /* file number is out of range. */
- return LIBMPQ_ERROR_EXIST;
- }
-
- /* return the compression status of file. */
- *compressed = (mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].flags & LIBMPQ_FLAG_COMPRESS_MULTI) != 0 ? TRUE : FALSE;
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-}
-
-/* this function return if the file is imploded or not. */
-int32_t libmpq__file_imploded(mpq_archive_s *mpq_archive, uint32_t file_number, uint32_t *imploded) {
-
- /* check if given file number is not out of range. */
- if (file_number < 0 || file_number > mpq_archive->files - 1) {
-
- /* file number is out of range. */
- return LIBMPQ_ERROR_EXIST;
- }
-
- /* return the implosion status of file. */
- *imploded = (mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].flags & LIBMPQ_FLAG_COMPRESS_PKZIP) != 0 ? TRUE : FALSE;
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-}
-
-/* this function return filenumber by the given name. */
-int32_t libmpq__file_number(mpq_archive_s *mpq_archive, const char *filename, uint32_t *number) {
-
- /* some common variables. */
- uint32_t i, hash1, hash2, hash3, ht_count;
-
- /* if the list of file names doesn't include this one, we'll have
- * to figure out the file number the "hard" way.
- */
- ht_count = mpq_archive->mpq_header.hash_table_count;
-
- hash1 = libmpq__hash_string (filename, 0x0) & (ht_count - 1);
- hash2 = libmpq__hash_string (filename, 0x100);
- hash3 = libmpq__hash_string (filename, 0x200);
-
- /* loop through all files in mpq archive.
- * hash1 gives us a clue about the starting position of this
- * search.
- */
- for (i = hash1; mpq_archive->mpq_hash[i].block_table_index != LIBMPQ_HASH_FREE; i = (i + 1) & (ht_count - 1)) {
-
- /* if the other two hashes match, we found our file number. */
- if (mpq_archive->mpq_hash[i].hash_a == hash2 &&
- mpq_archive->mpq_hash[i].hash_b == hash3) {
-
- /* return the file number. */
- *number = mpq_archive->mpq_hash[i].block_table_index - mpq_archive->mpq_map[mpq_archive->mpq_hash[i].block_table_index].block_table_diff;
-
- /* we found our file, return zero. */
- return LIBMPQ_SUCCESS;
- }
-
- /* check if we have cycled through the whole hash table */
- if (((i + 1) & (ht_count - 1)) == hash1) {
- break;
- }
- }
-
- /* if no matching entry found, so return error. */
- return LIBMPQ_ERROR_EXIST;
-}
-
-/* this function read the given file from archive into a buffer. */
-int32_t libmpq__file_read(mpq_archive_s *mpq_archive, uint32_t file_number, uint8_t *out_buf, libmpq__off_t out_size, libmpq__off_t *transferred) {
-
- /* some common variables. */
- uint32_t i;
- uint32_t blocks = 0;
- int32_t result = 0;
- libmpq__off_t file_offset = 0;
- libmpq__off_t unpacked_size = 0;
- libmpq__off_t transferred_block = 0;
- libmpq__off_t transferred_total = 0;
-
- /* check if given file number is not out of range. */
- if (file_number < 0 || file_number > mpq_archive->files - 1) {
-
- /* file number is out of range. */
- return LIBMPQ_ERROR_EXIST;
- }
-
- /* get target size of block. */
- libmpq__file_unpacked_size(mpq_archive, file_number, &unpacked_size);
-
- /* check if target buffer is to small. */
- if (unpacked_size > out_size) {
-
- /* output buffer size is to small or block size is unknown. */
- return LIBMPQ_ERROR_SIZE;
- }
-
- /* fetch file offset. */
- libmpq__file_offset(mpq_archive, file_number, &file_offset);
-
- /* get block count for file. */
- libmpq__file_blocks(mpq_archive, file_number, &blocks);
-
- /* open the packed block offset table. */
- if ((result = libmpq__block_open_offset(mpq_archive, file_number)) < 0) {
-
- /* something on opening packed block offset table failed. */
- return result;
- }
-
- /* loop through all blocks. */
- for (i = 0; i < blocks; i++) {
-
- /* cleanup size variable. */
- unpacked_size = 0;
-
- /* get unpacked block size. */
- libmpq__block_unpacked_size(mpq_archive, file_number, i, &unpacked_size);
-
- /* read block. */
- if ((result = libmpq__block_read(mpq_archive, file_number, i, out_buf + transferred_total, unpacked_size, &transferred_block)) < 0) {
-
- /* close the packed block offset table. */
- libmpq__block_close_offset(mpq_archive, file_number);
-
- /* something on reading block failed. */
- return result;
- }
-
- transferred_total += transferred_block;
-
- }
-
- /* close the packed block offset table. */
- libmpq__block_close_offset(mpq_archive, file_number);
-
- /* check for null pointer. */
- if (transferred != NULL) {
-
- /* store transferred bytes. */
- *transferred = transferred_total;
- }
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-}
-
-/* this function open a file in the given archive and caches the block offset information. */
-int32_t libmpq__block_open_offset(mpq_archive_s *mpq_archive, uint32_t file_number) {
-
- /* some common variables. */
- uint32_t i;
- uint32_t packed_size;
- int32_t rb = 0;
- int32_t result = 0;
-
- /* check if given file number is not out of range. */
- if (file_number < 0 || file_number > mpq_archive->files - 1) {
-
- /* file number is out of range. */
- return LIBMPQ_ERROR_EXIST;
- }
-
- if (mpq_archive->mpq_file[file_number]) {
-
- /* file already opened, so increment counter */
- mpq_archive->mpq_file[file_number]->open_count++;
- return LIBMPQ_SUCCESS;
- }
-
- /* check if file is not stored in a single sector. */
- if ((mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].flags & LIBMPQ_FLAG_SINGLE) == 0) {
-
- /* get packed size based on block size and block count. */
- packed_size = sizeof(uint32_t) * (((mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].unpacked_size + mpq_archive->block_size - 1) / mpq_archive->block_size) + 1);
- } else {
-
- /* file is stored in single sector and we need only two entries for the packed block offset table. */
- packed_size = sizeof(uint32_t) * 2;
- }
-
- /* check if data has one extra entry. */
- if ((mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].flags & LIBMPQ_FLAG_EXTRA) != 0) {
-
- /* add one uint32_t. */
- packed_size += sizeof(uint32_t);
- }
-
- /* allocate memory for the file. */
- if ((mpq_archive->mpq_file[file_number] = calloc(1, sizeof(mpq_file_s))) == NULL) {
-
- /* memory allocation problem. */
- result = LIBMPQ_ERROR_MALLOC;
- goto error;
- }
-
- /* allocate memory for the packed block offset table. */
- if ((mpq_archive->mpq_file[file_number]->packed_offset = calloc(1, packed_size)) == NULL) {
-
- /* memory allocation problem. */
- result = LIBMPQ_ERROR_MALLOC;
- goto error;
- }
-
- /* initialize counter to one opening */
- mpq_archive->mpq_file[file_number]->open_count = 1;
-
- /* check if we need to load the packed block offset table, we will maintain this table for unpacked files too. */
- if ((mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].flags & LIBMPQ_FLAG_COMPRESSED) != 0 &&
- (mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].flags & LIBMPQ_FLAG_SINGLE) == 0) {
-
- /* seek to block position. */
- if (fseeko(mpq_archive->fp, mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].offset + (((long long)mpq_archive->mpq_block_ex[mpq_archive->mpq_map[file_number].block_table_indices].offset_high) << 32) + mpq_archive->archive_offset, SEEK_SET) < 0) {
-
- /* seek in file failed. */
- result = LIBMPQ_ERROR_SEEK;
- goto error;
- }
-
- /* read block positions from begin of file. */
- if ((rb = fread(mpq_archive->mpq_file[file_number]->packed_offset, 1, packed_size, mpq_archive->fp)) < 0) {
-
- /* something on read from archive failed. */
- result = LIBMPQ_ERROR_READ;
- goto error;
- }
-
- /* check if the archive is protected some way, sometimes the file appears not to be encrypted, but it is. */
- if (mpq_archive->mpq_file[file_number]->packed_offset[0] != rb) {
-
- /* file is encrypted. */
- mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].flags |= LIBMPQ_FLAG_ENCRYPTED;
- }
-
- /* check if packed offset block is encrypted, we have to decrypt it. */
- if (mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].flags & LIBMPQ_FLAG_ENCRYPTED) {
-
- /* check if we don't know the file seed, try to find it. */
- if ((mpq_archive->mpq_file[file_number]->seed = libmpq__decrypt_key((uint8_t *)mpq_archive->mpq_file[file_number]->packed_offset, packed_size, mpq_archive->block_size)) < 0) {
-
- /* sorry without seed, we cannot extract file. */
- result = LIBMPQ_ERROR_DECRYPT;
- goto error;
- }
-
- /* decrypt block in input buffer. */
- if (libmpq__decrypt_block(mpq_archive->mpq_file[file_number]->packed_offset, packed_size, mpq_archive->mpq_file[file_number]->seed - 1) < 0 ) {
-
- /* something on decrypt failed. */
- result = LIBMPQ_ERROR_DECRYPT;
- goto error;
- }
-
- /* check if the block positions are correctly decrypted. */
- if (mpq_archive->mpq_file[file_number]->packed_offset[0] != packed_size) {
-
- /* sorry without seed, we cannot extract file. */
- result = LIBMPQ_ERROR_DECRYPT;
- goto error;
- }
- }
- } else {
-
- /* check if file is not stored in a single sector. */
- if ((mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].flags & LIBMPQ_FLAG_SINGLE) == 0) {
-
- /* loop through all blocks and create packed block offset table based on block size. */
- for (i = 0; i < ((mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].unpacked_size + mpq_archive->block_size - 1) / mpq_archive->block_size + 1); i++) {
-
- /* check if we process the last block. */
- if (i == ((mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].unpacked_size + mpq_archive->block_size - 1) / mpq_archive->block_size)) {
-
- /* store size of last block. */
- mpq_archive->mpq_file[file_number]->packed_offset[i] = mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].unpacked_size;
- } else {
-
- /* store default block size. */
- mpq_archive->mpq_file[file_number]->packed_offset[i] = i * mpq_archive->block_size;
- }
- }
- } else {
-
- /* store offsets. */
- mpq_archive->mpq_file[file_number]->packed_offset[0] = 0;
- mpq_archive->mpq_file[file_number]->packed_offset[1] = mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].packed_size;
- }
- }
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-
-error:
-
- /* free packed block offset table and file pointer. */
- free(mpq_archive->mpq_file[file_number]->packed_offset);
- free(mpq_archive->mpq_file[file_number]);
-
- /* return error constant. */
- return result;
-}
-
-/* this function free the file pointer to the opened file in archive. */
-int32_t libmpq__block_close_offset(mpq_archive_s *mpq_archive, uint32_t file_number) {
-
- /* check if given file number is not out of range. */
- if (file_number < 0 || file_number > mpq_archive->files - 1) {
-
- /* file number is out of range. */
- return LIBMPQ_ERROR_EXIST;
- }
-
- if (mpq_archive->mpq_file[file_number] == NULL) {
-
- /* packed block offset table is not opened. */
- return LIBMPQ_ERROR_OPEN;
- }
-
- mpq_archive->mpq_file[file_number]->open_count--;
-
- if (mpq_archive->mpq_file[file_number]->open_count != 0) {
-
- /* still in use */
- return LIBMPQ_SUCCESS;
- }
-
- /* free packed block offset table and file pointer. */
- free(mpq_archive->mpq_file[file_number]->packed_offset);
- free(mpq_archive->mpq_file[file_number]);
-
- /* mark it as unopened - libmpq__block_open_offset checks for this to decide whether to increment the counter */
- mpq_archive->mpq_file[file_number] = NULL;
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-}
-
-/* this function return the unpacked size of the given file and block in the archive. */
-int32_t libmpq__block_unpacked_size(mpq_archive_s *mpq_archive, uint32_t file_number, uint32_t block_number, libmpq__off_t *unpacked_size) {
-
- /* check if given file number is not out of range. */
- if (file_number < 0 || file_number > mpq_archive->files - 1) {
-
- /* file number is out of range. */
- return LIBMPQ_ERROR_EXIST;
- }
-
- /* check if given block number is not out of range. */
- if (block_number < 0 || block_number >= ((mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].flags & LIBMPQ_FLAG_SINGLE) != 0 ? 1 : (mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].unpacked_size + mpq_archive->block_size - 1) / mpq_archive->block_size)) {
-
- /* file number is out of range. */
- return LIBMPQ_ERROR_EXIST;
- }
-
- /* check if packed block offset table is opened. */
- if (mpq_archive->mpq_file[file_number] == NULL ||
- mpq_archive->mpq_file[file_number]->packed_offset == NULL) {
-
- /* packed block offset table is not opened. */
- return LIBMPQ_ERROR_OPEN;
- }
-
- /* check if block is stored as single sector. */
- if ((mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].flags & LIBMPQ_FLAG_SINGLE) != 0) {
-
- /* return the unpacked size of the block in the mpq archive. */
- *unpacked_size = mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].unpacked_size;
- }
-
- /* check if block is not stored as single sector. */
- if ((mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].flags & LIBMPQ_FLAG_SINGLE) == 0) {
-
- /* check if we not process the last block. */
- if (block_number < ((mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].unpacked_size + mpq_archive->block_size - 1) / mpq_archive->block_size) - 1) {
-
- /* return the block size as unpacked size. */
- *unpacked_size = mpq_archive->block_size;
- } else {
-
- /* return the unpacked size of the last block in the mpq archive. */
- *unpacked_size = mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].unpacked_size - mpq_archive->block_size * block_number;
- }
- }
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-}
-
-/* this function return the decryption seed for the given file and block. */
-int32_t libmpq__block_seed(mpq_archive_s *mpq_archive, uint32_t file_number, uint32_t block_number, uint32_t *seed) {
-
- /* check if given file number is not out of range. */
- if (file_number < 0 || file_number > mpq_archive->files - 1) {
-
- /* file number is out of range. */
- return LIBMPQ_ERROR_EXIST;
- }
-
- /* check if given block number is not out of range. */
- if (block_number < 0 || block_number >= ((mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].flags & LIBMPQ_FLAG_SINGLE) != 0 ? 1 : (mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].unpacked_size + mpq_archive->block_size - 1) / mpq_archive->block_size)) {
-
- /* file number is out of range. */
- return LIBMPQ_ERROR_EXIST;
- }
-
- /* check if packed block offset table is opened. */
- if (mpq_archive->mpq_file[file_number] == NULL ||
- mpq_archive->mpq_file[file_number]->packed_offset == NULL) {
-
- /* packed block offset table is not opened. */
- return LIBMPQ_ERROR_OPEN;
- }
-
- /* return the decryption key. */
- *seed = mpq_archive->mpq_file[file_number]->seed + block_number;
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-}
-
-/* this function read the given block from archive into a buffer. */
-int32_t libmpq__block_read(mpq_archive_s *mpq_archive, uint32_t file_number, uint32_t block_number, uint8_t *out_buf, libmpq__off_t out_size, libmpq__off_t *transferred) {
-
- /* some common variables. */
- uint8_t *in_buf;
- uint32_t seed = 0;
- uint32_t encrypted = 0;
- uint32_t compressed = 0;
- uint32_t imploded = 0;
- int32_t tb = 0;
- libmpq__off_t block_offset = 0;
- off_t in_size = 0;
- libmpq__off_t unpacked_size = 0;
-
- /* check if given file number is not out of range. */
- if (file_number < 0 || file_number > mpq_archive->files - 1) {
-
- /* file number is out of range. */
- return LIBMPQ_ERROR_EXIST;
- }
-
- /* check if given block number is not out of range. */
- if (block_number < 0 || block_number >= ((mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].flags & LIBMPQ_FLAG_SINGLE) != 0 ? 1 : (mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].unpacked_size + mpq_archive->block_size - 1) / mpq_archive->block_size)) {
-
- /* file number is out of range. */
- return LIBMPQ_ERROR_EXIST;
- }
-
- /* check if packed block offset table is opened. */
- if (mpq_archive->mpq_file[file_number] == NULL ||
- mpq_archive->mpq_file[file_number]->packed_offset == NULL) {
-
- /* packed block offset table is not opened. */
- return LIBMPQ_ERROR_OPEN;
- }
-
- /* get target size of block. */
- libmpq__block_unpacked_size(mpq_archive, file_number, block_number, &unpacked_size);
-
- /* check if target buffer is to small. */
- if (unpacked_size > out_size) {
-
- /* output buffer size is to small or block size is unknown. */
- return LIBMPQ_ERROR_SIZE;
- }
-
- /* fetch some required values like input buffer size and block offset. */
- block_offset = mpq_archive->mpq_block[mpq_archive->mpq_map[file_number].block_table_indices].offset + (((long long)mpq_archive->mpq_block_ex[mpq_archive->mpq_map[file_number].block_table_indices].offset_high) << 32) + mpq_archive->mpq_file[file_number]->packed_offset[block_number];
- in_size = mpq_archive->mpq_file[file_number]->packed_offset[block_number + 1] - mpq_archive->mpq_file[file_number]->packed_offset[block_number];
-
- /* seek in file. */
- if (fseeko(mpq_archive->fp, block_offset + mpq_archive->archive_offset, SEEK_SET) < 0) {
-
- /* something with seek in file failed. */
- return LIBMPQ_ERROR_SEEK;
- }
-
- /* allocate memory for the read buffer. */
- if ((in_buf = calloc(1, in_size)) == NULL) {
-
- /* memory allocation problem. */
- return LIBMPQ_ERROR_MALLOC;
- }
-
- /* read block from file. */
- if (fread(in_buf, 1, in_size, mpq_archive->fp) < 0) {
-
- /* free buffers. */
- free(in_buf);
-
- /* something on reading block failed. */
- return LIBMPQ_ERROR_READ;
- }
-
- /* get encryption status. */
- libmpq__file_encrypted(mpq_archive, file_number, &encrypted);
-
- /* check if file is encrypted. */
- if (encrypted == 1) {
-
- /* get decryption key. */
- libmpq__block_seed(mpq_archive, file_number, block_number, &seed);
-
- /* decrypt block. */
- if (libmpq__decrypt_block((uint32_t *)in_buf, in_size, seed) < 0) {
-
- /* free buffers. */
- free(in_buf);
-
- /* something on decrypting block failed. */
- return LIBMPQ_ERROR_DECRYPT;
- }
- }
-
- /* get compression status. */
- libmpq__file_compressed(mpq_archive, file_number, &compressed);
-
- /* check if file is compressed. */
- if (compressed == 1) {
-
- /* decompress block. */
- if ((tb = libmpq__decompress_block(in_buf, in_size, out_buf, out_size, LIBMPQ_FLAG_COMPRESS_MULTI)) < 0) {
-
- /* free temporary buffer. */
- free(in_buf);
-
- /* something on decompressing block failed. */
- return LIBMPQ_ERROR_UNPACK;
- }
- }
-
- /* get implosion status. */
- libmpq__file_imploded(mpq_archive, file_number, &imploded);
-
- /* check if file is imploded. */
- if (imploded == 1) {
-
- /* explode block. */
- if ((tb = libmpq__decompress_block(in_buf, in_size, out_buf, out_size, LIBMPQ_FLAG_COMPRESS_PKZIP)) < 0) {
-
- /* free temporary buffer. */
- free(in_buf);
-
- /* something on decompressing block failed. */
- return LIBMPQ_ERROR_UNPACK;
- }
- }
-
- /* check if file is neither compressed nor imploded. */
- if (compressed == 0 && imploded == 0) {
-
- /* copy block. */
- if ((tb = libmpq__decompress_block(in_buf, in_size, out_buf, out_size, LIBMPQ_FLAG_COMPRESS_NONE)) < 0) {
-
- /* free temporary buffer. */
- free(in_buf);
-
- /* something on decompressing block failed. */
- return LIBMPQ_ERROR_UNPACK;
- }
- }
-
- /* free read buffer. */
- free(in_buf);
-
- /* check for null pointer. */
- if (transferred != NULL) {
-
- /* store transferred bytes. */
- *transferred = tb;
- }
-
- /* if no error was found, return zero. */
- return LIBMPQ_SUCCESS;
-}