2010-01-11 20:41:07 +00:00
|
|
|
/* 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.
|
2014-02-18 01:34:18 +00:00
|
|
|
*
|
2010-01-11 20:41:07 +00:00
|
|
|
* 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.
|
2014-02-18 01:34:18 +00:00
|
|
|
*
|
2010-01-11 20:41:07 +00:00
|
|
|
* 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/scummsys.h"
|
|
|
|
#include "common/debug.h"
|
|
|
|
#include "common/util.h"
|
2010-05-09 18:27:56 +00:00
|
|
|
#include "common/file.h"
|
2010-05-10 18:23:54 +00:00
|
|
|
#include "common/fs.h"
|
2010-01-11 20:41:07 +00:00
|
|
|
#include "common/macresman.h"
|
2010-05-10 18:23:54 +00:00
|
|
|
#include "common/md5.h"
|
2010-11-19 17:03:07 +00:00
|
|
|
#include "common/substream.h"
|
2011-04-24 08:34:27 +00:00
|
|
|
#include "common/textconsole.h"
|
2016-03-01 18:14:50 +00:00
|
|
|
#include "common/archive.h"
|
2010-01-11 20:41:07 +00:00
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
#ifdef MACOSX
|
|
|
|
#include "common/config-manager.h"
|
|
|
|
#endif
|
2010-01-11 20:41:07 +00:00
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
namespace Common {
|
2010-01-11 20:41:07 +00:00
|
|
|
|
2010-05-10 18:23:54 +00:00
|
|
|
#define MBI_INFOHDR 128
|
|
|
|
#define MBI_ZERO1 0
|
|
|
|
#define MBI_NAMELEN 1
|
|
|
|
#define MBI_ZERO2 74
|
|
|
|
#define MBI_ZERO3 82
|
|
|
|
#define MBI_DFLEN 83
|
|
|
|
#define MBI_RFLEN 87
|
|
|
|
#define MAXNAMELEN 63
|
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
MacResManager::MacResManager() {
|
|
|
|
memset(this, 0, sizeof(MacResManager));
|
|
|
|
close();
|
2010-01-11 20:41:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
MacResManager::~MacResManager() {
|
2010-05-09 18:27:56 +00:00
|
|
|
close();
|
|
|
|
}
|
|
|
|
|
|
|
|
void MacResManager::close() {
|
|
|
|
_resForkOffset = -1;
|
|
|
|
_mode = kResForkNone;
|
|
|
|
|
2010-01-11 20:41:07 +00:00
|
|
|
for (int i = 0; i < _resMap.numTypes; i++) {
|
2010-05-12 20:22:10 +00:00
|
|
|
for (int j = 0; j < _resTypes[i].items; j++)
|
|
|
|
if (_resLists[i][j].nameOffset != -1)
|
|
|
|
delete[] _resLists[i][j].name;
|
|
|
|
|
|
|
|
delete[] _resLists[i];
|
2010-01-11 20:41:07 +00:00
|
|
|
}
|
|
|
|
|
2018-04-05 18:25:28 +00:00
|
|
|
delete[] _resLists; _resLists = nullptr;
|
|
|
|
delete[] _resTypes; _resTypes = nullptr;
|
|
|
|
delete _stream; _stream = nullptr;
|
2011-09-29 20:58:22 +00:00
|
|
|
_resMap.numTypes = 0;
|
2010-05-09 18:27:56 +00:00
|
|
|
}
|
|
|
|
|
2010-11-07 17:17:21 +00:00
|
|
|
bool MacResManager::hasDataFork() const {
|
2010-05-09 18:27:56 +00:00
|
|
|
return !_baseFileName.empty();
|
|
|
|
}
|
2010-01-11 20:41:07 +00:00
|
|
|
|
2010-11-07 17:17:21 +00:00
|
|
|
bool MacResManager::hasResFork() const {
|
2010-05-09 18:27:56 +00:00
|
|
|
return !_baseFileName.empty() && _mode != kResForkNone;
|
|
|
|
}
|
|
|
|
|
2011-02-04 15:27:56 +00:00
|
|
|
uint32 MacResManager::getResForkDataSize() const {
|
2010-05-10 18:23:54 +00:00
|
|
|
if (!hasResFork())
|
|
|
|
return 0;
|
|
|
|
|
2011-02-04 15:27:56 +00:00
|
|
|
_stream->seek(_resForkOffset + 4);
|
|
|
|
return _stream->readUint32BE();
|
2010-05-10 18:23:54 +00:00
|
|
|
}
|
|
|
|
|
2010-11-07 17:17:21 +00:00
|
|
|
String MacResManager::computeResForkMD5AsString(uint32 length) const {
|
2010-05-10 18:23:54 +00:00
|
|
|
if (!hasResFork())
|
2010-11-07 17:17:21 +00:00
|
|
|
return String();
|
2010-05-10 18:23:54 +00:00
|
|
|
|
2011-02-04 15:27:56 +00:00
|
|
|
_stream->seek(_resForkOffset);
|
|
|
|
uint32 dataOffset = _stream->readUint32BE() + _resForkOffset;
|
|
|
|
/* uint32 mapOffset = */ _stream->readUint32BE();
|
|
|
|
uint32 dataLength = _stream->readUint32BE();
|
|
|
|
|
|
|
|
|
|
|
|
SeekableSubReadStream resForkStream(_stream, dataOffset, dataOffset + dataLength);
|
2010-11-07 17:16:59 +00:00
|
|
|
return computeStreamMD5AsString(resForkStream, MIN<uint32>(length, _resForkSize));
|
2010-05-10 18:23:54 +00:00
|
|
|
}
|
|
|
|
|
2013-07-12 00:47:33 +00:00
|
|
|
bool MacResManager::open(const String &fileName) {
|
2010-05-09 18:27:56 +00:00
|
|
|
close();
|
|
|
|
|
|
|
|
#ifdef MACOSX
|
|
|
|
// Check the actual fork on a Mac computer
|
2013-07-12 00:47:33 +00:00
|
|
|
String fullPath = ConfMan.get("path") + "/" + fileName + "/..namedfork/rsrc";
|
2011-06-10 20:30:03 +00:00
|
|
|
FSNode resFsNode = FSNode(fullPath);
|
|
|
|
if (resFsNode.exists()) {
|
2011-10-29 09:23:44 +00:00
|
|
|
SeekableReadStream *macResForkRawStream = resFsNode.createReadStream();
|
2010-05-09 18:27:56 +00:00
|
|
|
|
2011-06-10 20:30:03 +00:00
|
|
|
if (macResForkRawStream && loadFromRawFork(*macResForkRawStream)) {
|
2013-07-12 00:47:33 +00:00
|
|
|
_baseFileName = fileName;
|
2011-06-10 20:30:03 +00:00
|
|
|
return true;
|
|
|
|
}
|
2010-09-17 03:55:41 +00:00
|
|
|
|
2011-06-10 20:30:03 +00:00
|
|
|
delete macResForkRawStream;
|
|
|
|
}
|
2010-05-09 18:27:56 +00:00
|
|
|
#endif
|
|
|
|
|
2010-11-07 17:17:21 +00:00
|
|
|
File *file = new File();
|
2010-05-09 18:27:56 +00:00
|
|
|
|
2013-07-12 00:47:33 +00:00
|
|
|
// Prefer standalone files first, starting with raw forks
|
|
|
|
if (file->open(fileName + ".rsrc") && loadFromRawFork(*file)) {
|
|
|
|
_baseFileName = fileName;
|
2010-05-09 18:27:56 +00:00
|
|
|
return true;
|
|
|
|
}
|
2010-05-20 13:46:18 +00:00
|
|
|
file->close();
|
2010-05-09 18:27:56 +00:00
|
|
|
|
2013-07-12 00:47:33 +00:00
|
|
|
// Then try for AppleDouble using Apple's naming
|
|
|
|
if (file->open(constructAppleDoubleName(fileName)) && loadFromAppleDouble(*file)) {
|
|
|
|
_baseFileName = fileName;
|
2010-05-09 18:27:56 +00:00
|
|
|
return true;
|
|
|
|
}
|
2010-05-20 13:46:18 +00:00
|
|
|
file->close();
|
|
|
|
|
2013-07-12 00:47:33 +00:00
|
|
|
// Check .bin for MacBinary next
|
|
|
|
if (file->open(fileName + ".bin") && loadFromMacBinary(*file)) {
|
|
|
|
_baseFileName = fileName;
|
2010-05-09 18:27:56 +00:00
|
|
|
return true;
|
|
|
|
}
|
2010-05-20 13:46:18 +00:00
|
|
|
file->close();
|
2010-05-09 18:27:56 +00:00
|
|
|
|
2013-07-12 00:47:33 +00:00
|
|
|
// As a last resort, see if just the data fork exists
|
|
|
|
if (file->open(fileName)) {
|
|
|
|
_baseFileName = fileName;
|
2010-05-18 10:39:08 +00:00
|
|
|
|
2013-07-12 00:47:33 +00:00
|
|
|
// FIXME: Is this really needed?
|
2010-05-18 10:39:08 +00:00
|
|
|
if (isMacBinary(*file)) {
|
2013-07-12 00:47:33 +00:00
|
|
|
file->seek(0);
|
2010-05-20 13:46:18 +00:00
|
|
|
if (loadFromMacBinary(*file))
|
|
|
|
return true;
|
2010-05-18 10:39:08 +00:00
|
|
|
}
|
2010-05-20 13:46:18 +00:00
|
|
|
|
2013-07-12 00:47:33 +00:00
|
|
|
file->seek(0);
|
2010-05-20 13:46:18 +00:00
|
|
|
_stream = file;
|
2010-05-09 18:27:56 +00:00
|
|
|
return true;
|
|
|
|
}
|
2010-05-20 13:46:18 +00:00
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
delete file;
|
|
|
|
|
|
|
|
// The file doesn't exist
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-07-12 00:47:33 +00:00
|
|
|
bool MacResManager::open(const FSNode &path, const String &fileName) {
|
2010-05-10 18:23:54 +00:00
|
|
|
close();
|
|
|
|
|
|
|
|
#ifdef MACOSX
|
|
|
|
// Check the actual fork on a Mac computer
|
2013-07-12 00:47:33 +00:00
|
|
|
String fullPath = path.getPath() + "/" + fileName + "/..namedfork/rsrc";
|
2011-06-10 20:30:03 +00:00
|
|
|
FSNode resFsNode = FSNode(fullPath);
|
|
|
|
if (resFsNode.exists()) {
|
2011-10-29 09:23:44 +00:00
|
|
|
SeekableReadStream *macResForkRawStream = resFsNode.createReadStream();
|
2010-05-10 18:23:54 +00:00
|
|
|
|
2011-06-10 20:30:03 +00:00
|
|
|
if (macResForkRawStream && loadFromRawFork(*macResForkRawStream)) {
|
2013-07-12 00:47:33 +00:00
|
|
|
_baseFileName = fileName;
|
2011-06-10 20:30:03 +00:00
|
|
|
return true;
|
|
|
|
}
|
2010-09-17 03:55:41 +00:00
|
|
|
|
2011-06-10 20:30:03 +00:00
|
|
|
delete macResForkRawStream;
|
|
|
|
}
|
2010-05-10 18:23:54 +00:00
|
|
|
#endif
|
|
|
|
|
2013-07-12 00:47:33 +00:00
|
|
|
// Prefer standalone files first, starting with raw forks
|
|
|
|
FSNode fsNode = path.getChild(fileName + ".rsrc");
|
2010-05-20 13:46:18 +00:00
|
|
|
if (fsNode.exists() && !fsNode.isDirectory()) {
|
|
|
|
SeekableReadStream *stream = fsNode.createReadStream();
|
2013-07-12 00:47:33 +00:00
|
|
|
if (loadFromRawFork(*stream)) {
|
|
|
|
_baseFileName = fileName;
|
2010-05-20 13:46:18 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
delete stream;
|
2010-05-10 18:23:54 +00:00
|
|
|
}
|
|
|
|
|
2013-07-12 00:47:33 +00:00
|
|
|
// Then try for AppleDouble using Apple's naming
|
|
|
|
fsNode = path.getChild(constructAppleDoubleName(fileName));
|
2010-05-20 13:46:18 +00:00
|
|
|
if (fsNode.exists() && !fsNode.isDirectory()) {
|
|
|
|
SeekableReadStream *stream = fsNode.createReadStream();
|
2013-07-12 00:47:33 +00:00
|
|
|
if (loadFromAppleDouble(*stream)) {
|
|
|
|
_baseFileName = fileName;
|
2010-05-20 13:46:18 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
delete stream;
|
2010-05-10 18:23:54 +00:00
|
|
|
}
|
2010-05-20 13:46:18 +00:00
|
|
|
|
2013-07-12 00:47:33 +00:00
|
|
|
// Check .bin for MacBinary next
|
|
|
|
fsNode = path.getChild(fileName + ".bin");
|
2010-05-20 13:46:18 +00:00
|
|
|
if (fsNode.exists() && !fsNode.isDirectory()) {
|
|
|
|
SeekableReadStream *stream = fsNode.createReadStream();
|
2013-07-12 00:47:33 +00:00
|
|
|
if (loadFromMacBinary(*stream)) {
|
|
|
|
_baseFileName = fileName;
|
2010-05-20 13:46:18 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
delete stream;
|
2010-05-10 18:23:54 +00:00
|
|
|
}
|
|
|
|
|
2013-07-12 00:47:33 +00:00
|
|
|
// As a last resort, see if just the data fork exists
|
|
|
|
fsNode = path.getChild(fileName);
|
2010-05-10 18:23:54 +00:00
|
|
|
if (fsNode.exists() && !fsNode.isDirectory()) {
|
2010-05-20 13:46:18 +00:00
|
|
|
SeekableReadStream *stream = fsNode.createReadStream();
|
2013-07-12 00:47:33 +00:00
|
|
|
_baseFileName = fileName;
|
2010-05-18 10:39:08 +00:00
|
|
|
|
2013-07-12 00:47:33 +00:00
|
|
|
// FIXME: Is this really needed?
|
2010-05-20 13:46:18 +00:00
|
|
|
if (isMacBinary(*stream)) {
|
2013-07-12 00:47:33 +00:00
|
|
|
stream->seek(0);
|
2010-05-20 13:46:18 +00:00
|
|
|
if (loadFromMacBinary(*stream))
|
|
|
|
return true;
|
2010-05-18 10:39:08 +00:00
|
|
|
}
|
2010-05-20 13:46:18 +00:00
|
|
|
|
2013-07-12 00:47:33 +00:00
|
|
|
stream->seek(0);
|
2010-05-20 13:46:18 +00:00
|
|
|
_stream = stream;
|
2010-05-10 18:23:54 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// The file doesn't exist
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-07-12 00:47:33 +00:00
|
|
|
bool MacResManager::exists(const String &fileName) {
|
2012-03-08 13:02:06 +00:00
|
|
|
// Try the file name by itself
|
2013-07-12 00:47:33 +00:00
|
|
|
if (File::exists(fileName))
|
2012-03-08 13:02:06 +00:00
|
|
|
return true;
|
|
|
|
|
|
|
|
// Try the .rsrc extension
|
2013-07-12 00:47:33 +00:00
|
|
|
if (File::exists(fileName + ".rsrc"))
|
2012-03-08 13:02:06 +00:00
|
|
|
return true;
|
|
|
|
|
|
|
|
// Check if we have a MacBinary file
|
2013-07-12 00:47:33 +00:00
|
|
|
File tempFile;
|
|
|
|
if (tempFile.open(fileName + ".bin") && isMacBinary(tempFile))
|
2012-03-08 13:02:06 +00:00
|
|
|
return true;
|
|
|
|
|
|
|
|
// Check if we have an AppleDouble file
|
2013-07-12 00:47:33 +00:00
|
|
|
if (tempFile.open(constructAppleDoubleName(fileName)) && tempFile.readUint32BE() == 0x00051607)
|
2012-03-08 13:02:06 +00:00
|
|
|
return true;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2016-03-01 18:17:35 +00:00
|
|
|
void MacResManager::listFiles(StringArray &files, const String &pattern) {
|
2016-03-01 18:14:50 +00:00
|
|
|
// Base names discovered so far.
|
|
|
|
typedef HashMap<String, bool, IgnoreCase_Hash, IgnoreCase_EqualTo> BaseNameSet;
|
|
|
|
BaseNameSet baseNames;
|
|
|
|
|
|
|
|
// List files itself.
|
|
|
|
ArchiveMemberList memberList;
|
|
|
|
SearchMan.listMatchingMembers(memberList, pattern);
|
|
|
|
SearchMan.listMatchingMembers(memberList, pattern + ".rsrc");
|
|
|
|
SearchMan.listMatchingMembers(memberList, pattern + ".bin");
|
|
|
|
SearchMan.listMatchingMembers(memberList, constructAppleDoubleName(pattern));
|
|
|
|
|
|
|
|
for (ArchiveMemberList::const_iterator i = memberList.begin(), end = memberList.end(); i != end; ++i) {
|
|
|
|
String filename = (*i)->getName();
|
|
|
|
|
|
|
|
// For raw resource forks and MacBinary files we strip the extension
|
|
|
|
// here to obtain a valid base name.
|
|
|
|
int lastDotPos = filename.size() - 1;
|
|
|
|
for (; lastDotPos >= 0; --lastDotPos) {
|
|
|
|
if (filename[lastDotPos] == '.') {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lastDotPos != -1) {
|
|
|
|
const char *extension = filename.c_str() + lastDotPos + 1;
|
|
|
|
bool removeExtension = false;
|
|
|
|
|
|
|
|
// TODO: Should we really keep filenames suggesting raw resource
|
|
|
|
// forks or MacBinary files but not being such around? This might
|
|
|
|
// depend on the pattern the client requests...
|
|
|
|
if (!scumm_stricmp(extension, "rsrc")) {
|
|
|
|
SeekableReadStream *stream = (*i)->createReadStream();
|
|
|
|
removeExtension = stream && isRawFork(*stream);
|
|
|
|
delete stream;
|
|
|
|
} else if (!scumm_stricmp(extension, "bin")) {
|
|
|
|
SeekableReadStream *stream = (*i)->createReadStream();
|
|
|
|
removeExtension = stream && isMacBinary(*stream);
|
|
|
|
delete stream;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (removeExtension) {
|
|
|
|
filename.erase(lastDotPos);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Strip AppleDouble '._' prefix if applicable.
|
|
|
|
bool isAppleDoubleName = false;
|
|
|
|
const String filenameAppleDoubleStripped = disassembleAppleDoubleName(filename, &isAppleDoubleName);
|
|
|
|
|
|
|
|
if (isAppleDoubleName) {
|
|
|
|
SeekableReadStream *stream = (*i)->createReadStream();
|
|
|
|
if (stream->readUint32BE() == 0x00051607) {
|
|
|
|
filename = filenameAppleDoubleStripped;
|
|
|
|
}
|
|
|
|
// TODO: Should we really keep filenames suggesting AppleDouble
|
|
|
|
// but not being AppleDouble around? This might depend on the
|
|
|
|
// pattern the client requests...
|
|
|
|
delete stream;
|
|
|
|
}
|
|
|
|
|
|
|
|
baseNames[filename] = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Append resulting base names to list to indicate found files.
|
|
|
|
for (BaseNameSet::const_iterator i = baseNames.begin(), end = baseNames.end(); i != end; ++i) {
|
|
|
|
files.push_back(i->_key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-11-07 17:17:21 +00:00
|
|
|
bool MacResManager::loadFromAppleDouble(SeekableReadStream &stream) {
|
2010-05-09 18:27:56 +00:00
|
|
|
if (stream.readUint32BE() != 0x00051607) // tag
|
|
|
|
return false;
|
|
|
|
|
|
|
|
stream.skip(20); // version + home file system
|
|
|
|
|
|
|
|
uint16 entryCount = stream.readUint16BE();
|
|
|
|
|
|
|
|
for (uint16 i = 0; i < entryCount; i++) {
|
|
|
|
uint32 id = stream.readUint32BE();
|
|
|
|
uint32 offset = stream.readUint32BE();
|
2010-05-10 18:23:54 +00:00
|
|
|
uint32 length = stream.readUint32BE(); // length
|
2010-05-09 18:27:56 +00:00
|
|
|
|
2010-12-05 00:51:04 +00:00
|
|
|
if (id == 2) {
|
|
|
|
// Found the resource fork!
|
2010-05-09 18:27:56 +00:00
|
|
|
_resForkOffset = offset;
|
|
|
|
_mode = kResForkAppleDouble;
|
2010-05-10 18:23:54 +00:00
|
|
|
_resForkSize = length;
|
2010-05-09 18:27:56 +00:00
|
|
|
return load(stream);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
2010-01-11 20:41:07 +00:00
|
|
|
}
|
|
|
|
|
2010-11-07 17:17:21 +00:00
|
|
|
bool MacResManager::isMacBinary(SeekableReadStream &stream) {
|
2010-05-18 10:39:08 +00:00
|
|
|
byte infoHeader[MBI_INFOHDR];
|
|
|
|
int resForkOffset = -1;
|
|
|
|
|
|
|
|
stream.read(infoHeader, MBI_INFOHDR);
|
|
|
|
|
|
|
|
if (infoHeader[MBI_ZERO1] == 0 && infoHeader[MBI_ZERO2] == 0 &&
|
|
|
|
infoHeader[MBI_ZERO3] == 0 && infoHeader[MBI_NAMELEN] <= MAXNAMELEN) {
|
|
|
|
|
|
|
|
// Pull out fork lengths
|
|
|
|
uint32 dataSize = READ_BE_UINT32(infoHeader + MBI_DFLEN);
|
|
|
|
uint32 rsrcSize = READ_BE_UINT32(infoHeader + MBI_RFLEN);
|
|
|
|
|
|
|
|
uint32 dataSizePad = (((dataSize + 127) >> 7) << 7);
|
|
|
|
uint32 rsrcSizePad = (((rsrcSize + 127) >> 7) << 7);
|
|
|
|
|
|
|
|
// Length check
|
|
|
|
if (MBI_INFOHDR + dataSizePad + rsrcSizePad == (uint32)stream.size()) {
|
|
|
|
resForkOffset = MBI_INFOHDR + dataSizePad;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (resForkOffset < 0)
|
|
|
|
return false;
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2016-03-01 18:14:50 +00:00
|
|
|
bool MacResManager::isRawFork(SeekableReadStream &stream) {
|
|
|
|
// TODO: Is there a better way to detect whether this is a raw fork?
|
|
|
|
const uint32 dataOffset = stream.readUint32BE();
|
|
|
|
const uint32 mapOffset = stream.readUint32BE();
|
|
|
|
const uint32 dataLength = stream.readUint32BE();
|
|
|
|
const uint32 mapLength = stream.readUint32BE();
|
|
|
|
|
|
|
|
return !stream.eos() && !stream.err()
|
|
|
|
&& dataOffset < (uint32)stream.size() && dataOffset + dataLength <= (uint32)stream.size()
|
|
|
|
&& mapOffset < (uint32)stream.size() && mapOffset + mapLength <= (uint32)stream.size();
|
|
|
|
}
|
|
|
|
|
2010-11-07 17:17:21 +00:00
|
|
|
bool MacResManager::loadFromMacBinary(SeekableReadStream &stream) {
|
2010-01-11 20:41:07 +00:00
|
|
|
byte infoHeader[MBI_INFOHDR];
|
2010-05-09 18:27:56 +00:00
|
|
|
stream.read(infoHeader, MBI_INFOHDR);
|
2010-01-11 20:41:07 +00:00
|
|
|
|
|
|
|
// Maybe we have MacBinary?
|
|
|
|
if (infoHeader[MBI_ZERO1] == 0 && infoHeader[MBI_ZERO2] == 0 &&
|
|
|
|
infoHeader[MBI_ZERO3] == 0 && infoHeader[MBI_NAMELEN] <= MAXNAMELEN) {
|
|
|
|
|
|
|
|
// Pull out fork lengths
|
2010-05-09 18:27:56 +00:00
|
|
|
uint32 dataSize = READ_BE_UINT32(infoHeader + MBI_DFLEN);
|
|
|
|
uint32 rsrcSize = READ_BE_UINT32(infoHeader + MBI_RFLEN);
|
2010-01-11 20:41:07 +00:00
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
uint32 dataSizePad = (((dataSize + 127) >> 7) << 7);
|
|
|
|
uint32 rsrcSizePad = (((rsrcSize + 127) >> 7) << 7);
|
2010-01-11 20:41:07 +00:00
|
|
|
|
|
|
|
// Length check
|
2010-05-10 18:23:54 +00:00
|
|
|
if (MBI_INFOHDR + dataSizePad + rsrcSizePad == (uint32)stream.size()) {
|
2010-05-09 18:27:56 +00:00
|
|
|
_resForkOffset = MBI_INFOHDR + dataSizePad;
|
2010-05-10 18:23:54 +00:00
|
|
|
_resForkSize = rsrcSize;
|
|
|
|
}
|
2010-01-11 20:41:07 +00:00
|
|
|
}
|
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
if (_resForkOffset < 0)
|
|
|
|
return false;
|
2010-01-11 20:41:07 +00:00
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
_mode = kResForkMacBinary;
|
|
|
|
return load(stream);
|
|
|
|
}
|
|
|
|
|
2010-11-07 17:17:21 +00:00
|
|
|
bool MacResManager::loadFromRawFork(SeekableReadStream &stream) {
|
2010-05-09 18:27:56 +00:00
|
|
|
_mode = kResForkRaw;
|
|
|
|
_resForkOffset = 0;
|
2010-05-10 18:23:54 +00:00
|
|
|
_resForkSize = stream.size();
|
2010-05-09 18:27:56 +00:00
|
|
|
return load(stream);
|
|
|
|
}
|
|
|
|
|
2010-11-07 17:17:21 +00:00
|
|
|
bool MacResManager::load(SeekableReadStream &stream) {
|
2010-05-09 18:27:56 +00:00
|
|
|
if (_mode == kResForkNone)
|
|
|
|
return false;
|
2010-01-11 20:41:07 +00:00
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
stream.seek(_resForkOffset);
|
|
|
|
|
|
|
|
_dataOffset = stream.readUint32BE() + _resForkOffset;
|
|
|
|
_mapOffset = stream.readUint32BE() + _resForkOffset;
|
|
|
|
_dataLength = stream.readUint32BE();
|
|
|
|
_mapLength = stream.readUint32BE();
|
2010-01-11 20:41:07 +00:00
|
|
|
|
|
|
|
// do sanity check
|
2012-12-13 05:02:22 +00:00
|
|
|
if (stream.eos() || _dataOffset >= (uint32)stream.size() || _mapOffset >= (uint32)stream.size() ||
|
|
|
|
_dataLength + _mapLength > (uint32)stream.size()) {
|
2010-05-09 18:27:56 +00:00
|
|
|
_resForkOffset = -1;
|
|
|
|
_mode = kResForkNone;
|
2010-01-11 20:41:07 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
debug(7, "got header: data %d [%d] map %d [%d]",
|
|
|
|
_dataOffset, _dataLength, _mapOffset, _mapLength);
|
2010-05-20 13:46:18 +00:00
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
_stream = &stream;
|
2010-01-11 20:41:07 +00:00
|
|
|
|
|
|
|
readMap();
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2010-11-07 17:17:21 +00:00
|
|
|
SeekableReadStream *MacResManager::getDataFork() {
|
2010-05-09 18:27:56 +00:00
|
|
|
if (!_stream)
|
2018-04-05 18:25:28 +00:00
|
|
|
return nullptr;
|
2010-05-09 18:27:56 +00:00
|
|
|
|
|
|
|
if (_mode == kResForkMacBinary) {
|
|
|
|
_stream->seek(MBI_DFLEN);
|
|
|
|
uint32 dataSize = _stream->readUint32BE();
|
2010-05-10 00:50:37 +00:00
|
|
|
return new SeekableSubReadStream(_stream, MBI_INFOHDR, MBI_INFOHDR + dataSize);
|
2010-05-09 18:27:56 +00:00
|
|
|
}
|
|
|
|
|
2010-11-07 17:17:21 +00:00
|
|
|
File *file = new File();
|
2010-05-09 18:27:56 +00:00
|
|
|
if (file->open(_baseFileName))
|
|
|
|
return file;
|
|
|
|
delete file;
|
|
|
|
|
2018-04-05 18:25:28 +00:00
|
|
|
return nullptr;
|
2010-05-09 18:27:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
MacResIDArray MacResManager::getResIDArray(uint32 typeID) {
|
2010-01-11 20:41:07 +00:00
|
|
|
int typeNum = -1;
|
|
|
|
MacResIDArray res;
|
|
|
|
|
|
|
|
for (int i = 0; i < _resMap.numTypes; i++)
|
2011-01-11 03:03:40 +00:00
|
|
|
if (_resTypes[i].id == typeID) {
|
2010-01-11 20:41:07 +00:00
|
|
|
typeNum = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeNum == -1)
|
|
|
|
return res;
|
2010-01-25 01:39:44 +00:00
|
|
|
|
2010-01-11 20:41:07 +00:00
|
|
|
res.resize(_resTypes[typeNum].items);
|
|
|
|
|
|
|
|
for (int i = 0; i < _resTypes[typeNum].items; i++)
|
|
|
|
res[i] = _resLists[typeNum][i].id;
|
|
|
|
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
2010-05-10 18:23:54 +00:00
|
|
|
MacResTagArray MacResManager::getResTagArray() {
|
|
|
|
MacResTagArray tagArray;
|
|
|
|
|
|
|
|
if (!hasResFork())
|
|
|
|
return tagArray;
|
|
|
|
|
|
|
|
tagArray.resize(_resMap.numTypes);
|
|
|
|
|
|
|
|
for (uint32 i = 0; i < _resMap.numTypes; i++)
|
|
|
|
tagArray[i] = _resTypes[i].id;
|
|
|
|
|
|
|
|
return tagArray;
|
|
|
|
}
|
|
|
|
|
2010-11-07 17:17:21 +00:00
|
|
|
String MacResManager::getResName(uint32 typeID, uint16 resID) const {
|
2010-01-11 20:41:07 +00:00
|
|
|
int typeNum = -1;
|
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
for (int i = 0; i < _resMap.numTypes; i++)
|
|
|
|
if (_resTypes[i].id == typeID) {
|
2010-01-11 20:41:07 +00:00
|
|
|
typeNum = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeNum == -1)
|
2010-05-09 18:27:56 +00:00
|
|
|
return "";
|
2010-01-11 20:41:07 +00:00
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
for (int i = 0; i < _resTypes[typeNum].items; i++)
|
2010-01-11 20:41:07 +00:00
|
|
|
if (_resLists[typeNum][i].id == resID)
|
|
|
|
return _resLists[typeNum][i].name;
|
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
return "";
|
2010-01-11 20:41:07 +00:00
|
|
|
}
|
|
|
|
|
2010-11-07 17:17:21 +00:00
|
|
|
SeekableReadStream *MacResManager::getResource(uint32 typeID, uint16 resID) {
|
2010-01-11 20:41:07 +00:00
|
|
|
int typeNum = -1;
|
|
|
|
int resNum = -1;
|
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
for (int i = 0; i < _resMap.numTypes; i++)
|
|
|
|
if (_resTypes[i].id == typeID) {
|
2010-01-11 20:41:07 +00:00
|
|
|
typeNum = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (typeNum == -1)
|
2018-04-05 18:25:28 +00:00
|
|
|
return nullptr;
|
2010-01-11 20:41:07 +00:00
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
for (int i = 0; i < _resTypes[typeNum].items; i++)
|
2010-01-11 20:41:07 +00:00
|
|
|
if (_resLists[typeNum][i].id == resID) {
|
|
|
|
resNum = i;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (resNum == -1)
|
2018-04-05 18:25:28 +00:00
|
|
|
return nullptr;
|
2010-01-11 20:41:07 +00:00
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
_stream->seek(_dataOffset + _resLists[typeNum][resNum].dataOffset);
|
|
|
|
uint32 len = _stream->readUint32BE();
|
2010-05-24 16:51:33 +00:00
|
|
|
|
|
|
|
// Ignore resources with 0 length
|
|
|
|
if (!len)
|
2018-04-05 18:25:28 +00:00
|
|
|
return nullptr;
|
2010-05-24 16:51:33 +00:00
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
return _stream->readStream(len);
|
2010-01-11 20:41:07 +00:00
|
|
|
}
|
|
|
|
|
2013-07-12 00:47:33 +00:00
|
|
|
SeekableReadStream *MacResManager::getResource(const String &fileName) {
|
2010-05-11 15:38:21 +00:00
|
|
|
for (uint32 i = 0; i < _resMap.numTypes; i++) {
|
2010-09-17 03:55:41 +00:00
|
|
|
for (uint32 j = 0; j < _resTypes[i].items; j++) {
|
2013-07-12 00:47:33 +00:00
|
|
|
if (_resLists[i][j].nameOffset != -1 && fileName.equalsIgnoreCase(_resLists[i][j].name)) {
|
2010-09-17 03:55:41 +00:00
|
|
|
_stream->seek(_dataOffset + _resLists[i][j].dataOffset);
|
|
|
|
uint32 len = _stream->readUint32BE();
|
|
|
|
|
|
|
|
// Ignore resources with 0 length
|
|
|
|
if (!len)
|
2018-04-05 18:25:28 +00:00
|
|
|
return nullptr;
|
2010-09-17 03:55:41 +00:00
|
|
|
|
|
|
|
return _stream->readStream(len);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-05 18:25:28 +00:00
|
|
|
return nullptr;
|
2010-09-17 03:55:41 +00:00
|
|
|
}
|
|
|
|
|
2013-07-12 00:47:33 +00:00
|
|
|
SeekableReadStream *MacResManager::getResource(uint32 typeID, const String &fileName) {
|
2010-09-17 03:55:41 +00:00
|
|
|
for (uint32 i = 0; i < _resMap.numTypes; i++) {
|
|
|
|
if (_resTypes[i].id != typeID)
|
|
|
|
continue;
|
|
|
|
|
2010-05-11 15:38:21 +00:00
|
|
|
for (uint32 j = 0; j < _resTypes[i].items; j++) {
|
2013-07-12 00:47:33 +00:00
|
|
|
if (_resLists[i][j].nameOffset != -1 && fileName.equalsIgnoreCase(_resLists[i][j].name)) {
|
2010-05-11 15:38:21 +00:00
|
|
|
_stream->seek(_dataOffset + _resLists[i][j].dataOffset);
|
|
|
|
uint32 len = _stream->readUint32BE();
|
2010-05-24 16:51:33 +00:00
|
|
|
|
|
|
|
// Ignore resources with 0 length
|
|
|
|
if (!len)
|
2018-04-05 18:25:28 +00:00
|
|
|
return nullptr;
|
2010-05-24 16:51:33 +00:00
|
|
|
|
2010-05-11 15:38:21 +00:00
|
|
|
return _stream->readStream(len);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-05 18:25:28 +00:00
|
|
|
return nullptr;
|
2010-05-11 15:38:21 +00:00
|
|
|
}
|
|
|
|
|
2010-01-11 20:41:07 +00:00
|
|
|
void MacResManager::readMap() {
|
2010-05-09 18:27:56 +00:00
|
|
|
_stream->seek(_mapOffset + 22);
|
2010-01-11 20:41:07 +00:00
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
_resMap.resAttr = _stream->readUint16BE();
|
|
|
|
_resMap.typeOffset = _stream->readUint16BE();
|
|
|
|
_resMap.nameOffset = _stream->readUint16BE();
|
|
|
|
_resMap.numTypes = _stream->readUint16BE();
|
2010-01-11 20:41:07 +00:00
|
|
|
_resMap.numTypes++;
|
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
_stream->seek(_mapOffset + _resMap.typeOffset + 2);
|
2010-01-11 20:41:07 +00:00
|
|
|
_resTypes = new ResType[_resMap.numTypes];
|
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
for (int i = 0; i < _resMap.numTypes; i++) {
|
|
|
|
_resTypes[i].id = _stream->readUint32BE();
|
|
|
|
_resTypes[i].items = _stream->readUint16BE();
|
|
|
|
_resTypes[i].offset = _stream->readUint16BE();
|
2010-01-11 20:41:07 +00:00
|
|
|
_resTypes[i].items++;
|
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
debug(8, "resType: <%s> items: %d offset: %d (0x%x)", tag2str(_resTypes[i].id), _resTypes[i].items, _resTypes[i].offset, _resTypes[i].offset);
|
2010-01-11 20:41:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
_resLists = new ResPtr[_resMap.numTypes];
|
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
for (int i = 0; i < _resMap.numTypes; i++) {
|
2010-01-11 20:41:07 +00:00
|
|
|
_resLists[i] = new Resource[_resTypes[i].items];
|
2010-05-09 18:27:56 +00:00
|
|
|
_stream->seek(_resTypes[i].offset + _mapOffset + _resMap.typeOffset);
|
2010-01-11 20:41:07 +00:00
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
for (int j = 0; j < _resTypes[i].items; j++) {
|
2010-01-11 20:41:07 +00:00
|
|
|
ResPtr resPtr = _resLists[i] + j;
|
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
resPtr->id = _stream->readUint16BE();
|
|
|
|
resPtr->nameOffset = _stream->readUint16BE();
|
|
|
|
resPtr->dataOffset = _stream->readUint32BE();
|
|
|
|
_stream->readUint32BE();
|
2018-04-05 18:25:28 +00:00
|
|
|
resPtr->name = nullptr;
|
2010-01-11 20:41:07 +00:00
|
|
|
|
|
|
|
resPtr->attr = resPtr->dataOffset >> 24;
|
|
|
|
resPtr->dataOffset &= 0xFFFFFF;
|
|
|
|
}
|
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
for (int j = 0; j < _resTypes[i].items; j++) {
|
2010-01-11 20:41:07 +00:00
|
|
|
if (_resLists[i][j].nameOffset != -1) {
|
2010-05-09 18:27:56 +00:00
|
|
|
_stream->seek(_resLists[i][j].nameOffset + _mapOffset + _resMap.nameOffset);
|
2010-01-11 20:41:07 +00:00
|
|
|
|
2010-05-09 18:27:56 +00:00
|
|
|
byte len = _stream->readByte();
|
2010-01-11 20:41:07 +00:00
|
|
|
_resLists[i][j].name = new char[len + 1];
|
|
|
|
_resLists[i][j].name[len] = 0;
|
2010-05-09 18:27:56 +00:00
|
|
|
_stream->read(_resLists[i][j].name, len);
|
2010-01-11 20:41:07 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-07-12 00:47:33 +00:00
|
|
|
String MacResManager::constructAppleDoubleName(String name) {
|
2012-04-08 01:19:11 +00:00
|
|
|
// Insert "._" before the last portion of a path name
|
|
|
|
for (int i = name.size() - 1; i >= 0; i--) {
|
|
|
|
if (i == 0) {
|
|
|
|
name.insertChar('_', 0);
|
|
|
|
name.insertChar('.', 0);
|
|
|
|
} else if (name[i] == '/') {
|
|
|
|
name.insertChar('_', i + 1);
|
|
|
|
name.insertChar('.', i + 1);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
2016-03-01 18:14:50 +00:00
|
|
|
String MacResManager::disassembleAppleDoubleName(String name, bool *isAppleDouble) {
|
|
|
|
if (isAppleDouble) {
|
|
|
|
*isAppleDouble = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Remove "._" before the last portion of a path name.
|
|
|
|
for (int i = name.size() - 1; i >= 0; --i) {
|
|
|
|
if (i == 0) {
|
|
|
|
if (name.size() > 2 && name[0] == '.' && name[1] == '_') {
|
|
|
|
name.erase(0, 2);
|
|
|
|
if (isAppleDouble) {
|
|
|
|
*isAppleDouble = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else if (name[i] == '/') {
|
|
|
|
if ((uint)(i + 2) < name.size() && name[i + 1] == '.' && name[i + 2] == '_') {
|
|
|
|
name.erase(i + 1, 2);
|
|
|
|
if (isAppleDouble) {
|
|
|
|
*isAppleDouble = true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
|
2010-01-11 20:41:07 +00:00
|
|
|
} // End of namespace Common
|