mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-07 10:21:31 +00:00
215 lines
4.2 KiB
C++
215 lines
4.2 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 3 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, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
|
|
#ifndef __FILE_H__
|
|
#define __FILE_H__
|
|
|
|
#include <stdio.h>
|
|
#include <stdlib.h>
|
|
|
|
#define FORBIDDEN_SYMBOL_ALLOW_ALL
|
|
|
|
#include "common/scummsys.h"
|
|
#include "common/endian.h"
|
|
#include "common/algorithm.h"
|
|
|
|
namespace Common {
|
|
|
|
enum AccessMode {
|
|
kFileReadMode = 1,
|
|
kFileWriteMode = 2
|
|
};
|
|
|
|
class File {
|
|
private:
|
|
::FILE *_f;
|
|
const byte *_memPtr;
|
|
size_t _offset, _size;
|
|
public:
|
|
File() : _f(nullptr), _memPtr(nullptr), _offset(0), _size(0) {}
|
|
|
|
bool open(const char *filename, AccessMode mode = kFileReadMode) {
|
|
_memPtr = nullptr;
|
|
_f = fopen(filename, (mode == kFileReadMode) ? "rb" : "wb+");
|
|
return (_f != NULL);
|
|
}
|
|
bool open(const byte *data, uint size_) {
|
|
close();
|
|
_f = nullptr;
|
|
_memPtr = data;
|
|
_size = size_;
|
|
return true;
|
|
}
|
|
|
|
void close() {
|
|
if (_f)
|
|
fclose(_f);
|
|
_f = nullptr;
|
|
delete[] _memPtr;
|
|
_memPtr = nullptr;
|
|
}
|
|
int seek(int offset, int whence = SEEK_SET) {
|
|
if (_f)
|
|
return fseek(_f, offset, whence);
|
|
|
|
switch (whence) {
|
|
case SEEK_SET:
|
|
_offset = offset;
|
|
break;
|
|
case SEEK_CUR:
|
|
_offset += offset;
|
|
break;
|
|
case SEEK_END:
|
|
_offset = _size + offset;
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
|
|
return _offset;
|
|
}
|
|
void skip(int offset) {
|
|
if (_f)
|
|
fseek(_f, offset, SEEK_CUR);
|
|
else
|
|
_offset += offset;
|
|
}
|
|
long read(void *buffer, size_t len) {
|
|
if (_f)
|
|
return fread(buffer, 1, len, _f);
|
|
|
|
uint bytesToRead = CLIP(len, (size_t)0, _size - _offset);
|
|
memcpy(buffer, &_memPtr[_offset], bytesToRead);
|
|
_offset += bytesToRead;
|
|
return bytesToRead;
|
|
}
|
|
void write(const void *buffer, size_t len) {
|
|
assert(_f);
|
|
fwrite(buffer, 1, len, _f);
|
|
}
|
|
void write(File &src, size_t len) {
|
|
for (size_t idx = 0; idx < len; ++idx)
|
|
writeByte(src.readByte());
|
|
}
|
|
byte readByte() {
|
|
byte v;
|
|
read(&v, sizeof(byte));
|
|
return v;
|
|
}
|
|
uint16 readWord() {
|
|
uint16 v;
|
|
read(&v, sizeof(uint16));
|
|
return FROM_LE_16(v);
|
|
}
|
|
uint readLong() {
|
|
uint v;
|
|
read(&v, sizeof(uint));
|
|
return FROM_LE_32(v);
|
|
}
|
|
|
|
uint readUint16BE() {
|
|
uint16 v;
|
|
read(&v, sizeof(uint16));
|
|
return FROM_BE_16(v);
|
|
}
|
|
uint readUint16LE() {
|
|
uint16 v;
|
|
read(&v, sizeof(uint16));
|
|
return FROM_LE_16(v);
|
|
}
|
|
uint readUint32BE() {
|
|
uint32 v;
|
|
read(&v, sizeof(uint32));
|
|
return FROM_BE_32(v);
|
|
}
|
|
uint readUint32LE() {
|
|
uint32 v;
|
|
read(&v, sizeof(uint32));
|
|
return FROM_LE_32(v);
|
|
}
|
|
|
|
void writeByte(byte v) {
|
|
write(&v, sizeof(byte));
|
|
}
|
|
void writeByte(byte v, int len) {
|
|
byte *b = new byte[len];
|
|
memset(b, v, len);
|
|
write(b, len);
|
|
delete[] b;
|
|
}
|
|
void writeWord(uint16 v) {
|
|
uint16 vTemp = TO_LE_16(v);
|
|
write(&vTemp, sizeof(uint16));
|
|
}
|
|
void writeLong(uint v) {
|
|
uint vTemp = TO_LE_32(v);
|
|
write(&vTemp, sizeof(uint));
|
|
}
|
|
void writeString(const char *msg) {
|
|
if (!msg) {
|
|
writeByte(0);
|
|
} else {
|
|
do {
|
|
writeByte(*msg);
|
|
} while (*msg++);
|
|
}
|
|
}
|
|
void writeString(File &src) {
|
|
char c;
|
|
do {
|
|
c = src.readByte();
|
|
writeByte(c);
|
|
} while (c);
|
|
}
|
|
uint pos() const {
|
|
if (_f)
|
|
return ftell(_f);
|
|
else
|
|
return _offset;
|
|
}
|
|
uint size() const {
|
|
if (_f) {
|
|
uint currentPos = pos();
|
|
fseek(_f, 0, SEEK_END);
|
|
uint result = pos();
|
|
fseek(_f, currentPos, SEEK_SET);
|
|
return result;
|
|
} else if (_memPtr) {
|
|
return _size;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
bool eof() const {
|
|
if (_f)
|
|
return feof(_f) != 0;
|
|
else if (_memPtr)
|
|
return _offset >= _size;
|
|
return false;
|
|
}
|
|
};
|
|
|
|
}
|
|
|
|
extern Common::File inputFile, outputFile;
|
|
|
|
#endif
|