scummvm/engines/tinsel/drives.cpp
2021-12-26 18:48:43 +01:00

245 lines
5.3 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/>.
*
* CD/drive handling functions
*/
#include "common/textconsole.h"
#include "tinsel/drives.h"
#include "tinsel/scene.h"
#include "tinsel/tinsel.h"
#include "tinsel/sched.h"
#include "tinsel/strres.h"
namespace Tinsel {
// These vars are reset upon engine destruction
char g_currentCD = '1';
static bool g_bChangingCD = false;
static char g_nextCD = '\0';
static uint32 g_lastTime = 0;
extern LANGUAGE g_sampleLanguage;
void ResetVarsDrives() {
g_currentCD = '1';
g_bChangingCD = false;
g_nextCD = '\0';
g_lastTime = 0;
}
void CdCD(CORO_PARAM) {
CORO_BEGIN_CONTEXT;
CORO_END_CONTEXT(_ctx);
CORO_BEGIN_CODE(_ctx);
while (g_bChangingCD) {
if (CoroScheduler.getCurrentProcess()) {
// FIXME: CdCD gets passed a Common::nullContext in RegisterGlobals() and
// PrimeSceneHopper(), because I didn't know how to get a proper
// context without converting the whole calling stack to CORO'd
// functions. If these functions really get called while a CD
// change is requested, this needs to be resolved.
if (coroParam == Common::nullContext)
error("CdCD needs context");
CORO_SLEEP(1);
} else
error("No current process in CdCD()");
}
CORO_END_CODE;
}
int GetCurrentCD() {
// count from 1
return (g_currentCD - '1' + 1);
}
static const uint32 cdFlags[] = { fCd1, fCd2, fCd3, fCd4, fCd5, fCd6, fCd7, fCd8 };
void SetCD(int flags) {
if (flags & cdFlags[g_currentCD - '1'])
return;
error("SetCD() problem");
}
int GetCD(int flags) {
int i;
char cd = '\0';
if (flags & cdFlags[g_currentCD - '1'])
return GetCurrentCD();
for (i = 0; i < 8; i++) {
if (flags & cdFlags[i]) {
cd = (char)(i + '1');
break;
}
}
assert(i != 8);
g_nextCD = cd;
return cd;
}
void DoCdChange() {
if (g_bChangingCD && (g_system->getMillis() > (g_lastTime + 1000))) {
g_lastTime = g_system->getMillis();
_vm->_sound->closeSampleStream();
// Use the filesize of the sample file to determine, for Discworld 2, which CD it is
if (TinselV2) {
TinselFile f;
if (!f.open(_vm->getSampleFile(g_sampleLanguage)))
// No CD present
return;
char sampleCdNumber = (f.size() >= (200 * 1024 * 1024)) ? '1' : '2';
f.close();
if (g_currentCD != sampleCdNumber)
return;
}
_vm->_sound->openSampleFiles();
ChangeLanguage(TextLanguage());
g_bChangingCD = false;
}
}
void SetNextCD(int cdNumber) {
assert(cdNumber == 1 || cdNumber == 2);
g_nextCD = (char)(cdNumber + '1' - 1);
}
bool GotoCD() {
// WORKAROUND: Somehow, CdDoChange() is called twice... Hopefully, this guard helps
if (g_currentCD == g_nextCD)
return false;
g_currentCD = g_nextCD;
/* if (bNoCD) {
strcpy(cdDirectory, hdDirectory);
cdLastBit[3] = currentCD;
strcat(cdDirectory, cdLastBit);
}
*/
g_bChangingCD = true;
return true;
}
bool TinselFile::_warningShown = false;
TinselFile::TinselFile() : ReadStreamEndian(TinselV1Saturn) {
_stream = nullptr;
}
TinselFile::TinselFile(bool bigEndian) : ReadStreamEndian(bigEndian) {
_stream = nullptr;
}
TinselFile::~TinselFile() {
delete _stream;
}
bool TinselFile::openInternal(const Common::String &filename) {
_stream = SearchMan.createReadStreamForMember(filename);
if (!_stream)
_stream = SearchMan.createReadStreamForMember(filename + ".");
return _stream != 0;
}
bool TinselFile::open(const Common::String &filename) {
if (openInternal(filename))
return true;
if (!TinselV2)
return false;
// Check if the file being requested is the *1.* or *2.* files
const char *fname = filename.c_str();
const char *p = strchr(fname, '1');
if (!p)
p = strchr(fname, '2');
if (!p || (*(p + 1) != '.'))
return false;
// Form a filename without the CD number character
char newFilename[50];
strncpy(newFilename, fname, p - fname);
strcpy(newFilename + (p - fname), p + 1);
return openInternal(newFilename);
}
void TinselFile::close() {
delete _stream;
_stream = nullptr;
}
int64 TinselFile::pos() const {
assert(_stream);
return _stream->pos();
}
int64 TinselFile::size() const {
assert(_stream);
return _stream->size();
}
bool TinselFile::seek(int64 offset, int whence) {
assert(_stream);
return _stream->seek(offset, whence);
}
bool TinselFile::eos() const {
assert(_stream);
return _stream->eos();
}
bool TinselFile::err() const {
assert(_stream);
return _stream->err();
}
void TinselFile::clearErr() {
assert(_stream);
_stream->clearErr();
}
uint32 TinselFile::read(void *dataPtr, uint32 dataSize) {
assert(_stream);
return _stream->read(dataPtr, dataSize);
}
} // End of namespace Tinsel