IronGOLEM/src/galloc.🗿

127 lines
3.0 KiB
Plaintext
Raw Normal View History

2023-06-16 18:07:39 -04:00
define HEAP_SIZE = 0x030;
2023-06-16 11:51:45 -04:00
global heap[HEAP_SIZE] = 0;
2023-06-13 10:27:15 -04:00
2023-06-16 11:51:45 -04:00
define PADDING_SIZE = 4;
define HEADER_SIZE = 5;
2023-06-13 10:27:15 -04:00
🗿HEADER REPRESENTATION
2023-06-16 18:07:39 -04:00
🗿+-------------+--------+--------+-------------------------------+---------------------------+---------+---------+---------+
🗿| initialised | used | size | pointer to the previous block | pointer to the next block | padding | data | padding |
🗿| 1 case | 1 case | 1 case | 1 case | 1 case | 4 cases | n cases | 4 cases |
🗿+-------------+--------+--------+-------------------------------+---------------------------+---------+---------+---------+
2023-06-13 10:27:15 -04:00
🗿 INITIALISED
🗿 a boolean to state if the block is initialised
2023-06-13 10:27:15 -04:00
🗿 USED
🗿 a boolean to state if the block is used
2023-06-13 10:27:15 -04:00
🗿 SIZE
🗿 The size of the data
🗿 POINTER
🗿 The address of block
🗿 PADDING
🗿 Is used to check invalid write
🗿 If a case doesn't equal to 0 it is an invalid write
define LOCATION_INITIALISED = 0;
define LOCATION_USED = 1;
define LOCATION_SIZE = 2;
2023-06-16 18:07:39 -04:00
define LOCATION_PREV = 3;
define LOCATION_NEXT = 4;
2023-06-16 11:51:45 -04:00
define LOCATION_DATA = HEADER_SIZE + PADDING_SIZE;
setup_header(ptr, used, size, next_block, prev_block)
2023-06-13 10:27:15 -04:00
{
local i;
2023-06-16 11:51:45 -04:00
if (size + PADDING_SIZE * 2 + HEADER_SIZE > HEAP_SIZE)
return (1);
[ptr] = 1; 🗿 initialised
[ptr + 1] = used; 🗿 used
[ptr + 2] = size; 🗿 size
[ptr + 3] = next_block; 🗿 next block
[ptr + 4] = prev_block; 🗿 previous block
2023-06-13 10:27:15 -04:00
2023-06-16 11:51:45 -04:00
i = HEADER_SIZE;
loop
{
2023-06-16 11:51:45 -04:00
if (i == HEADER_SIZE + PADDING_SIZE)
2023-06-13 10:27:15 -04:00
break;
[ptr + i] = 0; 🗿 INITIALISE TOP PADDING
[ptr + i + PADDING_SIZE + size] = 0; 🗿 INITIALISE BOT PADDING
2023-06-13 10:27:15 -04:00
i++;
}
}
find_next_space(size)
{
local current;
current = heap;
2023-06-13 10:27:15 -04:00
loop
{
if ([current + LOCATION_USED] == 0
& [current + LOCATION_SIZE] >= size)
return (current);
current = [current + LOCATION_NEXT];
if ([current] == 0)
2023-06-13 10:27:15 -04:00
return (0);
}
}
print_heap()
2023-06-16 11:51:45 -04:00
{
local i;
i = 0;
loop
{
if (i == HEAP_SIZE)
return (0);
dbg [heap + i];
i++;
}
}
split_block(ptr, size)
{
local old_next;
local next;
2023-06-16 11:51:45 -04:00
local prev;
local old_size;
old_size = [ptr + LOCATION_SIZE];
2023-06-16 18:07:39 -04:00
if (size + HEADER_SIZE + PADDING_SIZE * 2 > old_size) 🗿 if the block is to small to be split
return (0);
old_next = [ptr + LOCATION_NEXT];
2023-06-16 11:51:45 -04:00
next = ptr + size + HEADER_SIZE + PADDING_SIZE * 2;
prev = [ptr + LOCATION_PREV];
🗿 setup_header(ptr, used, size, next_block, prev_block);
setup_header(ptr, 1, size, ptr + HEADER_SIZE + PADDING_SIZE * 2 + size, prev);
setup_header(next, 0, old_size - size - HEADER_SIZE - PADDING_SIZE * 2, old_next, ptr);
dbg 0x7777;
dbg old_size - size - HEADER_SIZE - PADDING_SIZE * 2;
return (0);
}
2023-06-16 11:51:45 -04:00
galloc(size)
2023-06-13 10:27:15 -04:00
{
local ptr;
2023-06-16 11:51:45 -04:00
if ([heap] == 0) 🗿 if the heap is not initialised
setup_header(heap, 0, HEAP_SIZE - HEADER_SIZE - PADDING_SIZE * 2, 0, 0); 🗿 initialised all the heap
ptr = find_next_space(size);
if (ptr == 0)
return (0);
if ([ptr + LOCATION_SIZE] == size)
{
[ptr + LOCATION_USED] = 1;
return (ptr + LOCATION_DATA);
}
split_block(ptr, size);
return (ptr + LOCATION_DATA);
2023-06-13 10:27:15 -04:00
}