mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-06 09:48:39 +00:00
033ff78ae9
svn-id: r12176
188 lines
4.6 KiB
C++
188 lines
4.6 KiB
C++
/* ScummVM - Scumm Interpreter
|
|
* Copyright (C) 2002-2004 The ScummVM project
|
|
*
|
|
* 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.
|
|
*
|
|
* $Header$
|
|
*
|
|
*/
|
|
|
|
#include "stdafx.h"
|
|
#include "common/util.h"
|
|
#include "common/savefile.h"
|
|
|
|
#ifdef USE_ZLIB
|
|
#include <zlib.h>
|
|
#endif
|
|
|
|
uint32 SaveFile::read(void *ptr, uint32 size) {
|
|
return fread(ptr, 1, size);
|
|
}
|
|
|
|
byte SaveFile::readByte() {
|
|
byte b;
|
|
// TODO: Proper error handling
|
|
if (fread(&b, 1, 1) != 1)
|
|
return 0;
|
|
return b;
|
|
}
|
|
|
|
uint16 SaveFile::readUint16LE() {
|
|
uint16 a = readByte();
|
|
uint16 b = readByte();
|
|
return a | (b << 8);
|
|
}
|
|
|
|
uint32 SaveFile::readUint32LE() {
|
|
uint32 a = readUint16LE();
|
|
uint32 b = readUint16LE();
|
|
return (b << 16) | a;
|
|
}
|
|
|
|
uint16 SaveFile::readUint16BE() {
|
|
uint16 b = readByte();
|
|
uint16 a = readByte();
|
|
return a | (b << 8);
|
|
}
|
|
|
|
uint32 SaveFile::readUint32BE() {
|
|
uint32 b = readUint16BE();
|
|
uint32 a = readUint16BE();
|
|
return (b << 16) | a;
|
|
}
|
|
|
|
uint32 SaveFile::write(const void *ptr, uint32 size) {
|
|
return fwrite(ptr, 1, size);
|
|
}
|
|
|
|
void SaveFile::writeByte(byte value) {
|
|
fwrite(&value, 1, 1);
|
|
}
|
|
|
|
void SaveFile::writeUint16LE(uint16 value) {
|
|
writeByte((byte)(value & 0xff));
|
|
writeByte((byte)(value >> 8));
|
|
}
|
|
|
|
void SaveFile::writeUint32LE(uint32 value) {
|
|
writeUint16LE((uint16)(value & 0xffff));
|
|
writeUint16LE((uint16)(value >> 16));
|
|
}
|
|
|
|
void SaveFile::writeUint16BE(uint16 value) {
|
|
writeByte((byte)(value >> 8));
|
|
writeByte((byte)(value & 0xff));
|
|
}
|
|
|
|
void SaveFile::writeUint32BE(uint32 value) {
|
|
writeUint16BE((uint16)(value >> 16));
|
|
writeUint16BE((uint16)(value & 0xffff));
|
|
}
|
|
|
|
|
|
class StdioSaveFile : public SaveFile {
|
|
private:
|
|
FILE *fh;
|
|
public:
|
|
StdioSaveFile(const char *filename, bool saveOrLoad)
|
|
{ fh = ::fopen(filename, (saveOrLoad? "wb" : "rb")); }
|
|
~StdioSaveFile()
|
|
{ if(fh) ::fclose(fh); }
|
|
|
|
bool isOpen() const { return fh != 0; }
|
|
|
|
protected:
|
|
int fread(void *buf, int size, int cnt)
|
|
{ return ::fread(buf, size, cnt, fh); }
|
|
int fwrite(const void *buf, int size, int cnt)
|
|
{ return ::fwrite(buf, size, cnt, fh); }
|
|
};
|
|
|
|
|
|
#ifdef USE_ZLIB
|
|
class GzipSaveFile : public SaveFile {
|
|
private:
|
|
gzFile fh;
|
|
public:
|
|
GzipSaveFile(const char *filename, bool saveOrLoad)
|
|
{ fh = ::gzopen(filename, (saveOrLoad? "wb" : "rb")); }
|
|
~GzipSaveFile()
|
|
{ if(fh) ::gzclose(fh); }
|
|
|
|
bool isOpen() const { return fh != 0; }
|
|
|
|
protected:
|
|
int fread(void *buf, int size, int cnt) {
|
|
return ::gzread(fh, buf, size * cnt);
|
|
}
|
|
int fwrite(const void *buf, int size, int cnt) {
|
|
// Due to a "bug" in the zlib headers (or maybe I should say,
|
|
// a bug in the C++ spec? Whatever <g>) we have to be a bit
|
|
// hackish here and remove the const qualifier.
|
|
// Note that gzwrite's buf param is declared as "const voidp"
|
|
// which you might think is the same as "const void *" but it
|
|
// is not - rather it is equal to "void const *" which is the
|
|
// same as "void *". Hrmpf
|
|
return ::gzwrite(fh, const_cast<void *>(buf), size * cnt);
|
|
}
|
|
};
|
|
#endif
|
|
|
|
|
|
SaveFile *SaveFileManager::open_savefile(const char *filename, const char *directory, bool saveOrLoad) {
|
|
char buf[256];
|
|
join_paths(filename, directory, buf, sizeof(buf));
|
|
SaveFile *sf = makeSaveFile(buf, saveOrLoad);
|
|
if (!sf->isOpen()) {
|
|
delete sf;
|
|
sf = 0;
|
|
}
|
|
return sf;
|
|
}
|
|
|
|
void SaveFileManager::join_paths(const char *filename, const char *directory,
|
|
char *buf, int bufsize) {
|
|
buf[bufsize-1] = '\0';
|
|
strncpy(buf, directory, bufsize-1);
|
|
|
|
#ifdef WIN32
|
|
// Fix for Win98 issue related with game directory pointing to root drive ex. "c:\"
|
|
if ((buf[0] != 0) && (buf[1] == ':') && (buf[2] == '\\') && (buf[3] == 0)) {
|
|
buf[2] = 0;
|
|
}
|
|
#endif
|
|
|
|
const int dirLen = strlen(buf);
|
|
|
|
if (dirLen > 0) {
|
|
#ifdef __MORPHOS__
|
|
if (buf[dirLen-1] != ':' && buf[dirLen-1] != '/')
|
|
#endif
|
|
|
|
#if !defined(__GP32__) && !defined(__PALM_OS__)
|
|
strncat(buf, "/", bufsize-1); // prevent double /
|
|
#endif
|
|
}
|
|
strncat(buf, filename, bufsize-1);
|
|
}
|
|
|
|
SaveFile *SaveFileManager::makeSaveFile(const char *filename, bool saveOrLoad) {
|
|
#ifdef USE_ZLIB
|
|
return new GzipSaveFile(filename, saveOrLoad);
|
|
#else
|
|
return new StdioSaveFile(filename, saveOrLoad);
|
|
#endif
|
|
}
|