mirror of
https://github.com/libretro/scummvm.git
synced 2025-01-10 11:51:52 +00:00
185 lines
4.4 KiB
C++
185 lines
4.4 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
|
|
* aint32 with this program; if not, write to the Free Software
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA.
|
|
*
|
|
*
|
|
* Based on the original sources
|
|
* Faery Tale II -- The Halls of the Dead
|
|
* (c) 1993-1996 The Wyrmkeep Entertainment Co.
|
|
*/
|
|
|
|
#ifndef SAGA2_ONCALL_H
|
|
#define SAGA2_ONCALL_H
|
|
|
|
#include "saga2/bitarray.h"
|
|
|
|
namespace Saga2 {
|
|
|
|
#define isValidPtr(p) ((p!=NULL)&&(p!=(void *)0xCDCDCDCD))
|
|
|
|
class HandleArray {
|
|
private:
|
|
Common::Array<byte*> _handles;
|
|
uint32 _tileID;
|
|
byte*(*_loader)(hResID, bool);
|
|
public:
|
|
HandleArray(uint16 size, byte*(*loadfunction)(hResID, bool), uint32 newID) {
|
|
for (int i = 0; i < size; ++i)
|
|
_handles.push_back(nullptr);
|
|
_loader = loadfunction;
|
|
_tileID = newID;
|
|
}
|
|
|
|
void flush() {
|
|
for (int i = 0; i < _handles.size(); ++i) {
|
|
if (_handles[i]) {
|
|
free(_handles[i]);
|
|
_handles[i] = nullptr;
|
|
}
|
|
}
|
|
}
|
|
|
|
byte *operator[](uint32 ind) {
|
|
if (_handles[ind])
|
|
return _handles[ind];
|
|
|
|
return _handles[ind] = _loader(_tileID + MKTAG(0, 0, 0, ind), false);
|
|
}
|
|
};
|
|
|
|
template <class RESTYPE> class LoadOnCall {
|
|
BitArray wanted;
|
|
BitArray locked;
|
|
BitArray recent;
|
|
RESTYPE *handle;
|
|
uint16 handles;
|
|
uint32 tileID;
|
|
RHANDLE(*loader)(hResID, bool);
|
|
|
|
RESTYPE rLoad(uint16 ind, bool asynch);
|
|
void rFree(uint16);
|
|
void rInit(uint16);
|
|
|
|
public:
|
|
|
|
LoadOnCall(uint16 newSize, RHANDLE(*loadfunction)(hResID, bool), uint32 newID) {
|
|
tileID = newID;
|
|
wanted.resize(newSize);
|
|
locked.resize(newSize);
|
|
recent.resize(newSize);
|
|
handle = (RESTYPE *)malloc(sizeof(RESTYPE) * newSize);
|
|
handles = newSize;
|
|
loader = loadfunction;
|
|
for (uint16 i = 0; i < newSize; i++)
|
|
rInit(i);
|
|
}
|
|
|
|
// destructor
|
|
~LoadOnCall() {
|
|
//if (handles) delete[] handle;
|
|
}
|
|
|
|
RESTYPE operator[](uint32 ind) {
|
|
// assert (ind<handles);
|
|
if (!locked[ind]) handle[ind] = rLoad(ind, false);
|
|
return handle[ind];
|
|
}
|
|
|
|
void flush(void) {
|
|
//recent.clearAll();
|
|
for (uint16 i = 0; i < handles; i++)
|
|
if ((locked[i]) && (!recent[i]))
|
|
rFree(i);
|
|
//locked|=recent;
|
|
recent.clearAll();
|
|
}
|
|
|
|
BitArray lockState(void) {
|
|
return locked;
|
|
}
|
|
|
|
BitArray wantState(void) {
|
|
return wanted;
|
|
}
|
|
|
|
};
|
|
|
|
template <class RESTYPE> RESTYPE LoadOnCall<RESTYPE>::rLoad(uint16 ind, bool asynch) {
|
|
RESTYPE t;
|
|
if (isValidPtr(handle[ind]) && locked[ind]) {
|
|
RLockHandle((RHANDLE) handle[ind]);
|
|
locked.Bit(ind, true);
|
|
wanted.Bit(ind, false);
|
|
recent.Bit(ind, true);
|
|
return handle[ind];
|
|
}
|
|
if (isValidPtr(handle[ind]) && wanted[ind]) {
|
|
// wait for handle
|
|
RLockHandle((RHANDLE) handle[ind]);
|
|
locked.Bit(ind, true);
|
|
wanted.Bit(ind, false);
|
|
recent.Bit(ind, true);
|
|
return NULL;
|
|
}
|
|
|
|
//t = (RESTYPE) tileRes->load( tileID + MKTAG( 0,0,0,bankNum ) );
|
|
t = (RESTYPE) loader(tileID + MKTAG(0, 0, 0, ind), asynch);
|
|
|
|
locked.Bit(ind, !asynch);
|
|
wanted.Bit(ind, asynch);
|
|
recent.Bit(ind, true);
|
|
|
|
if (asynch) {
|
|
handle[ind] = (RESTYPE) NULL;
|
|
} else {
|
|
if (t == NULL) {
|
|
error("Resource %d could not load (Tile bank)", ind);
|
|
}
|
|
handle[ind] = t;
|
|
}
|
|
return t;
|
|
}
|
|
|
|
template <class RESTYPE> void LoadOnCall<RESTYPE>::rFree(uint16 ind) {
|
|
if (isValidPtr(handle[ind])) {
|
|
RUnlockHandle((RHANDLE) handle[ind]);
|
|
locked.Bit(ind, false);
|
|
//recent.Bit(ind,false);
|
|
// washHandle(handle[ind]);
|
|
}
|
|
}
|
|
|
|
template <class RESTYPE> void LoadOnCall<RESTYPE>::rInit(uint16 ind) {
|
|
RESTYPE t;
|
|
if (!isValidPtr(handle[ind])) {
|
|
t = (RESTYPE) loader(tileID + MKTAG(0, 0, 0, ind), false);
|
|
handle[ind] = t;
|
|
locked.Bit(ind, true);
|
|
RUnlockHandle((RHANDLE) handle[ind]);
|
|
locked.Bit(ind, false);
|
|
recent.Bit(ind, false);
|
|
}
|
|
}
|
|
|
|
//typedef LoadOnCall<TileBankHandle> tileBankArray; //(64,tileResLoad);
|
|
|
|
} // end of namespace Saga2
|
|
|
|
#endif
|