mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-27 05:32:45 +00:00
241 lines
6.5 KiB
C++
241 lines
6.5 KiB
C++
/* ScummVM - Graphic Adventure Engine
|
|
*
|
|
* ScummVM is the legal property of its developers, whose names
|
|
* are too numerous to list here. Please refer to the COPYRIGHT
|
|
* file distributed with this source distribution.
|
|
*
|
|
* 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*
|
|
*/
|
|
|
|
#include "common/file.h"
|
|
#include "common/debug.h"
|
|
|
|
#include "sludge/debug.h"
|
|
#include "sludge/stringy.h"
|
|
#include "sludge/allfiles.h"
|
|
#include "sludge/moreio.h"
|
|
#include "sludge/newfatal.h"
|
|
#include "sludge/sludge.h"
|
|
#include "sludge/CommonCode/version.h"
|
|
|
|
namespace Sludge {
|
|
|
|
bool sliceBusy = true;
|
|
|
|
Common::File *bigDataFile = NULL;
|
|
|
|
uint32 startOfDataIndex, startOfTextIndex, startOfSubIndex, startOfObjectIndex;
|
|
|
|
bool openSubSlice(int num) {
|
|
// FILE * dbug = fopen ("debuggy.txt", "at");
|
|
|
|
// fprintf (dbug, "\nTrying to open sub %i\n", num);
|
|
|
|
if (sliceBusy) {
|
|
fatal("Can't read from data file", "I'm already reading something");
|
|
return false;
|
|
}
|
|
// fprintf (dbug, "Going to position %li\n", startOfSubIndex + (num << 2));
|
|
bigDataFile->seek(startOfSubIndex + (num << 2), 0);
|
|
bigDataFile->seek(bigDataFile->readUint32LE(), 0);
|
|
// fprintf (dbug, "Told to skip forward to %li\n", ftell (bigDataFile));
|
|
// fclose (dbug);
|
|
|
|
return sliceBusy = true;
|
|
}
|
|
|
|
bool openObjectSlice(int num) {
|
|
// FILE * dbug = fopen ("debuggy.txt", "at");
|
|
|
|
// fprintf (dbug, "\nTrying to open object %i\n", num);
|
|
|
|
if (sliceBusy) {
|
|
fatal("Can't read from data file", "I'm already reading something");
|
|
return false;
|
|
}
|
|
|
|
// fprintf (dbug, "Going to position %li\n", startOfObjectIndex + (num << 2));
|
|
bigDataFile->seek(startOfObjectIndex + (num << 2), 0);
|
|
bigDataFile->seek(bigDataFile->readUint32LE(), 0);
|
|
// fprintf (dbug, "Told to skip forward to %li\n", ftell (bigDataFile));
|
|
// fclose (dbug);
|
|
return sliceBusy = true;
|
|
}
|
|
|
|
unsigned int openFileFromNum(int num) {
|
|
// FILE * dbug = fopen ("debuggy.txt", "at");
|
|
|
|
if (sliceBusy) {
|
|
fatal("Can't read from data file", "I'm already reading something");
|
|
return 0;
|
|
}
|
|
|
|
// fprintf (dbug, "\nTrying to open file %i\n", num);
|
|
// fprintf (dbug, "Jumping to %li (for index) \n", startOfDataIndex + (num << 2));
|
|
bigDataFile->seek(startOfDataIndex + (num << 2), 0);
|
|
bigDataFile->seek(bigDataFile->readUint32LE(), 1);
|
|
// fprintf (dbug, "Jumping to %li (for data) \n", ftell (bigDataFile));
|
|
sliceBusy = true;
|
|
// fclose (dbug);
|
|
|
|
return bigDataFile->readUint32LE();
|
|
}
|
|
|
|
// Converts a string from Windows CP-1252 to UTF-8.
|
|
// This is needed for old games.
|
|
char *convertString(char *s) {
|
|
#if 0
|
|
static char *buf = NULL;
|
|
|
|
if (! buf) {
|
|
buf = new char [65536];
|
|
if (! checkNew(buf)) return NULL;
|
|
}
|
|
|
|
char **tmp1 = (char **) &s;
|
|
char **tmp2 = (char **) &buf;
|
|
char *sOrig = s;
|
|
char *bufOrig = buf;
|
|
#if defined __unix__ && !(defined __APPLE__)
|
|
iconv_t convert = iconv_open("UTF-8", "ISO8859-2");
|
|
#else
|
|
iconv_t convert = iconv_open("UTF-8", "CP1250");
|
|
#endif
|
|
|
|
if (convert == (iconv_t) - 1) {
|
|
switch (errno) {
|
|
case EINVAL:
|
|
fprintf(stderr, "Error: Encoding not supported by iconv.\n");
|
|
break;
|
|
default:
|
|
fprintf(stderr, "Error: Could not open iconv conversion descriptor.\n");
|
|
}
|
|
}
|
|
|
|
size_t len1 = strlen(s) + 1;
|
|
size_t len2 = 65535;
|
|
size_t iconv_value =
|
|
#ifdef _WIN32
|
|
iconv(convert, (const char **) tmp1, &len1, tmp2, &len2);
|
|
#else
|
|
iconv(convert, (char **) tmp1, &len1, tmp2, &len2);
|
|
#endif
|
|
|
|
if (iconv_value == (size_t) - 1) {
|
|
switch (errno) {
|
|
/* See "man 3 iconv" for an explanation. */
|
|
case EILSEQ:
|
|
fprintf(stderr, "Invalid multibyte sequence.\n");
|
|
break;
|
|
case EINVAL:
|
|
fprintf(stderr, "Incomplete multibyte sequence.\n");
|
|
break;
|
|
case E2BIG:
|
|
fprintf(stderr, "No more room.\n");
|
|
break;
|
|
default:
|
|
fprintf(stderr, "Error: %s.\n", strerror(errno));
|
|
}
|
|
fatal("Conversion to Unicode failed. This can be fixed by recompiling the game in a current version of the SLUDGE Development Kit, but this error should never happen. Congratulations, you've found a bug in the SLUDGE engine! Please let us know about it.");
|
|
}
|
|
iconv_close(convert);
|
|
|
|
delete [] sOrig;
|
|
return copyString(buf = bufOrig);
|
|
#endif
|
|
return s; //TODO: false value
|
|
}
|
|
|
|
char *getNumberedString(int value) {
|
|
|
|
if (sliceBusy) {
|
|
fatal("Can't read from data file", "I'm already reading something");
|
|
return NULL;
|
|
}
|
|
|
|
bigDataFile->seek((value << 2) + startOfTextIndex, 0);
|
|
value = bigDataFile->readUint32LE();
|
|
bigDataFile->seek(value, 0);
|
|
|
|
char *s = readString(bigDataFile);
|
|
|
|
if (gameVersion < VERSION(2, 2)) {
|
|
// This is an older game - We need to convert the string to UTF-8
|
|
s = convertString(s);
|
|
}
|
|
|
|
return s;
|
|
}
|
|
|
|
bool startAccess() {
|
|
int wasBusy = sliceBusy;
|
|
sliceBusy = true;
|
|
return wasBusy;
|
|
}
|
|
void finishAccess() {
|
|
sliceBusy = false;
|
|
}
|
|
|
|
int32_t startIndex;
|
|
|
|
void setFileIndices(Common::File *fp, int numLanguages, unsigned int skipBefore) {
|
|
if (fp) {
|
|
// Keep hold of the file handle, and let things get at it
|
|
bigDataFile = fp;
|
|
startIndex = fp->pos();
|
|
} else {
|
|
// No file pointer - this means that we reuse the bigDataFile
|
|
fp = bigDataFile;
|
|
fp->seek(startIndex, SEEK_SET);
|
|
}
|
|
sliceBusy = false;
|
|
|
|
if (skipBefore > numLanguages) {
|
|
warning("Not a valid language ID! Using default instead.");
|
|
skipBefore = 0;
|
|
}
|
|
|
|
// STRINGS
|
|
int skipAfter = numLanguages - skipBefore;
|
|
while (skipBefore) {
|
|
fp->seek(fp->readUint32LE(), SEEK_SET);
|
|
skipBefore--;
|
|
}
|
|
startOfTextIndex = fp->pos() + 4;
|
|
debug(kSludgeDebugDataLoad, "startOfTextIndex: %i", startOfTextIndex);
|
|
|
|
fp->seek(fp->readUint32LE(), SEEK_SET);
|
|
|
|
while (skipAfter) {
|
|
fp->seek(fp->readUint32LE(), SEEK_SET);
|
|
skipAfter--;
|
|
}
|
|
|
|
startOfSubIndex = fp->pos() + 4;
|
|
fp->seek(fp->readUint32LE(), SEEK_CUR);
|
|
debug(kSludgeDebugDataLoad, "startOfSubIndex: %i", startOfTextIndex);
|
|
|
|
startOfObjectIndex = fp->pos() + 4;
|
|
fp->seek(fp->readUint32LE(), SEEK_CUR);
|
|
debug(kSludgeDebugDataLoad, "startOfObjectIndex: %i", startOfTextIndex);
|
|
|
|
// Remember that the data section starts here
|
|
startOfDataIndex = fp->pos();
|
|
debug(kSludgeDebugDataLoad, "startOfDataIndex: %i", startOfTextIndex);
|
|
}
|
|
|
|
} // End of namespace Sludge
|