2012-11-01 15:19:01 +00:00
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "../../Globals.h"
|
|
|
|
|
|
|
|
#include <vector>
|
|
|
|
#include <list>
|
2012-11-03 02:33:24 +00:00
|
|
|
#include <cstring>
|
2012-11-01 15:19:01 +00:00
|
|
|
|
|
|
|
|
|
|
|
// Generic allocator thingy
|
|
|
|
// Allocates blocks from a range
|
|
|
|
|
|
|
|
class BlockAllocator
|
|
|
|
{
|
|
|
|
public:
|
2012-11-07 16:35:22 +00:00
|
|
|
BlockAllocator(int grain = 16); // 16 byte granularity by default.
|
2012-11-01 15:19:01 +00:00
|
|
|
~BlockAllocator();
|
|
|
|
|
|
|
|
void Init(u32 _rangeStart, u32 _rangeSize);
|
|
|
|
void Shutdown();
|
|
|
|
|
|
|
|
void ListBlocks();
|
|
|
|
|
|
|
|
// WARNING: size can be modified upwards!
|
2012-11-05 11:02:09 +00:00
|
|
|
u32 Alloc(u32 &size, bool fromTop = false, const char *tag = 0);
|
2012-11-01 15:19:01 +00:00
|
|
|
u32 AllocAt(u32 position, u32 size, const char *tag = 0);
|
|
|
|
|
2012-11-07 18:10:34 +00:00
|
|
|
bool Free(u32 position);
|
2012-11-01 15:19:01 +00:00
|
|
|
bool IsBlockFree(u32 position) {
|
|
|
|
Block *b = GetBlockFromAddress(position);
|
|
|
|
if (b)
|
|
|
|
return !b->taken;
|
|
|
|
else
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
void MergeFreeBlocks();
|
|
|
|
|
|
|
|
u32 GetBlockStartFromAddress(u32 addr);
|
|
|
|
u32 GetBlockSizeFromAddress(u32 addr);
|
|
|
|
u32 GetLargestFreeBlockSize();
|
|
|
|
u32 GetTotalFreeBytes();
|
2012-11-05 11:02:09 +00:00
|
|
|
|
|
|
|
private:
|
2012-11-07 16:35:22 +00:00
|
|
|
void CheckBlocks();
|
|
|
|
|
2012-11-05 11:02:09 +00:00
|
|
|
struct Block
|
|
|
|
{
|
|
|
|
Block(u32 _start, u32 _size, bool _taken) : start(_start), size(_size), taken(_taken)
|
|
|
|
{
|
2012-11-05 12:36:12 +00:00
|
|
|
strcpy(tag, "(untitled)");
|
2012-11-05 11:02:09 +00:00
|
|
|
}
|
|
|
|
void SetTag(const char *_tag) {
|
|
|
|
if (_tag)
|
2012-11-07 18:10:34 +00:00
|
|
|
strncpy(tag, _tag, 32);
|
2012-11-05 11:02:09 +00:00
|
|
|
else
|
2012-11-07 18:10:34 +00:00
|
|
|
strncpy(tag, "---", 32);
|
|
|
|
tag[31] = 0;
|
2012-11-05 11:02:09 +00:00
|
|
|
}
|
|
|
|
u32 start;
|
|
|
|
u32 size;
|
|
|
|
bool taken;
|
2012-11-07 18:10:34 +00:00
|
|
|
char tag[32];
|
2012-11-05 11:02:09 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
std::list<Block> blocks;
|
2012-11-05 12:36:12 +00:00
|
|
|
u32 rangeStart_;
|
|
|
|
u32 rangeSize_;
|
2012-11-05 11:02:09 +00:00
|
|
|
|
2012-11-07 16:35:22 +00:00
|
|
|
u32 grain_;
|
|
|
|
|
2012-11-05 11:02:09 +00:00
|
|
|
Block *GetBlockFromAddress(u32 addr);
|
|
|
|
std::list<Block>::iterator GetBlockIterFromAddress(u32 addr);
|
2012-11-01 15:19:01 +00:00
|
|
|
};
|