2003-12-16 02:10:15 +00:00
|
|
|
/* ScummVM - Scumm Interpreter
|
2004-01-06 12:45:34 +00:00
|
|
|
* Copyright (C) 2003-2004 The ScummVM project
|
2003-12-16 02:10:15 +00:00
|
|
|
*
|
|
|
|
* 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 "memman.h"
|
|
|
|
#include "common/util.h"
|
|
|
|
|
2004-01-11 15:47:41 +00:00
|
|
|
namespace Sword1 {
|
|
|
|
|
2003-12-16 02:10:15 +00:00
|
|
|
MemMan::MemMan(void) {
|
|
|
|
_alloced = 0;
|
|
|
|
_memListFree = _memListFreeEnd = NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
MemMan::~MemMan(void) {
|
|
|
|
}
|
|
|
|
|
2004-01-11 15:47:41 +00:00
|
|
|
void MemMan::alloc(MemHandle *bsMem, uint32 pSize, uint16 pCond) {
|
2003-12-16 02:10:15 +00:00
|
|
|
_alloced += pSize;
|
|
|
|
bsMem->data = (void*)malloc(pSize);
|
|
|
|
if (!bsMem->data)
|
|
|
|
error("MemMan::alloc(): Can't alloc %d bytes of memory.", pSize);
|
|
|
|
bsMem->cond = pCond;
|
|
|
|
bsMem->size = pSize;
|
|
|
|
if (pCond == MEM_CAN_FREE) {
|
|
|
|
warning("%d Bytes alloced as FREEABLE.", pSize); // why should one want to alloc mem if it can be freed?
|
|
|
|
addToFreeList(bsMem);
|
|
|
|
} else if (bsMem->next || bsMem->prev) // it's in our _freeAble list, remove it from there
|
2003-12-19 14:07:12 +00:00
|
|
|
removeFromFreeList(bsMem);
|
2003-12-16 02:10:15 +00:00
|
|
|
checkMemoryUsage();
|
|
|
|
}
|
|
|
|
|
2004-01-11 15:47:41 +00:00
|
|
|
void MemMan::freeNow(MemHandle *bsMem) {
|
2003-12-16 02:10:15 +00:00
|
|
|
if (bsMem->cond != MEM_FREED) {
|
|
|
|
_alloced -= bsMem->size;
|
|
|
|
removeFromFreeList(bsMem);
|
|
|
|
free(bsMem->data);
|
|
|
|
bsMem->cond = MEM_FREED;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-01-11 15:47:41 +00:00
|
|
|
void MemMan::setCondition(MemHandle *bsMem, uint16 pCond) {
|
2003-12-16 02:10:15 +00:00
|
|
|
if ((pCond == MEM_FREED) || (pCond > MEM_DONT_FREE))
|
|
|
|
error("MemMan::setCondition: program tried to set illegal memory condition");
|
|
|
|
if (bsMem->cond != pCond) {
|
|
|
|
bsMem->cond = pCond;
|
|
|
|
if (pCond == MEM_DONT_FREE)
|
|
|
|
removeFromFreeList(bsMem);
|
|
|
|
else if (pCond == MEM_CAN_FREE)
|
|
|
|
addToFreeList(bsMem);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2003-12-20 09:12:54 +00:00
|
|
|
void MemMan::flush(void) {
|
|
|
|
while (_memListFree) {
|
|
|
|
free(_memListFreeEnd->data);
|
|
|
|
_memListFreeEnd->data = NULL;
|
|
|
|
_memListFreeEnd->cond = MEM_FREED;
|
|
|
|
_alloced -= _memListFreeEnd->size;
|
|
|
|
removeFromFreeList(_memListFreeEnd);
|
|
|
|
}
|
|
|
|
if (_alloced)
|
|
|
|
warning("MemMan::flush: Something's wrong: still %d bytes alloced", _alloced);
|
|
|
|
}
|
|
|
|
|
2003-12-16 02:10:15 +00:00
|
|
|
void MemMan::checkMemoryUsage(void) {
|
|
|
|
while ((_alloced > MAX_ALLOC) && _memListFree) {
|
|
|
|
free(_memListFreeEnd->data);
|
|
|
|
_memListFreeEnd->data = NULL;
|
|
|
|
_memListFreeEnd->cond = MEM_FREED;
|
|
|
|
_alloced -= _memListFreeEnd->size;
|
|
|
|
removeFromFreeList(_memListFreeEnd);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2004-01-11 15:47:41 +00:00
|
|
|
void MemMan::addToFreeList(MemHandle *bsMem) {
|
2003-12-16 02:10:15 +00:00
|
|
|
if (bsMem->next || bsMem->prev) {
|
|
|
|
warning("addToFreeList: mem block is already in freeList");
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
bsMem->prev = NULL;
|
|
|
|
bsMem->next = _memListFree;
|
|
|
|
if (bsMem->next)
|
|
|
|
bsMem->next->prev = bsMem;
|
|
|
|
_memListFree = bsMem;
|
|
|
|
if (!_memListFreeEnd)
|
|
|
|
_memListFreeEnd = _memListFree;
|
|
|
|
}
|
|
|
|
|
2004-01-11 15:47:41 +00:00
|
|
|
void MemMan::removeFromFreeList(MemHandle *bsMem) {
|
2003-12-16 02:10:15 +00:00
|
|
|
if (_memListFree == bsMem)
|
|
|
|
_memListFree = bsMem->next;
|
|
|
|
if (_memListFreeEnd == bsMem)
|
|
|
|
_memListFreeEnd = bsMem->prev;
|
|
|
|
|
|
|
|
if (bsMem->next)
|
|
|
|
bsMem->next->prev = bsMem->prev;
|
|
|
|
if (bsMem->prev)
|
|
|
|
bsMem->prev->next = bsMem->next;
|
|
|
|
bsMem->next = bsMem->prev = NULL;
|
|
|
|
}
|
|
|
|
|
2004-01-11 15:47:41 +00:00
|
|
|
void MemMan::initHandle(MemHandle *bsMem) {
|
|
|
|
memset(bsMem, 0, sizeof(MemHandle));
|
2003-12-16 02:10:15 +00:00
|
|
|
}
|
|
|
|
|
2004-01-11 15:47:41 +00:00
|
|
|
} // End of namespace Sword1
|