rpn/src/stack.hpp

320 lines
8.1 KiB
C++
Raw Normal View History

#ifndef __stack_h__
#define __stack_h__
2014-02-12 11:26:26 +01:00
#include <string.h>
#include <map>
using namespace std;
// allocation base size
#define ALLOC_STACK_CHUNK (64*1024)
//
class stack
{
public:
2015-05-19 17:51:03 +02:00
stack()
{
2017-04-29 15:43:05 +02:00
_base = NULL;
2017-05-22 18:22:15 +02:00
_base_pointer = NULL;
2017-04-29 15:43:05 +02:00
_total_size = 0;
2017-05-22 18:22:15 +02:00
_total_count_pointer = 0;
2017-04-23 15:49:28 +02:00
erase();
}
2015-05-19 17:51:03 +02:00
virtual ~stack()
2015-05-19 17:51:03 +02:00
{
2017-04-29 15:43:05 +02:00
if (_base != NULL)
free(_base);
2017-05-22 18:22:15 +02:00
if (_base_pointer != NULL)
free(_base_pointer);
}
2017-04-23 15:49:28 +02:00
void erase()
{
_current = _base;
_count = 0;
}
2017-05-02 10:47:03 +02:00
//
static void copy_and_push_back(stack& from, unsigned int index_from, stack& to)
{
2017-05-02 10:47:03 +02:00
// copy a whole stack entry and push it back to another stack
object* allocated = to.allocate_back(from.seq_len(index_from), from.seq_type(index_from));
memcpy(allocated, from.seq_obj(index_from), from.seq_len(index_from));
2017-05-02 10:47:03 +02:00
if (allocated->_type == cmd_number)
2017-06-14 22:17:49 +02:00
((number*)allocated)->move();
else if (allocated->_type == cmd_complex)
((complex*)allocated)->move();
}
2015-05-19 17:51:03 +02:00
//
2017-05-02 10:47:03 +02:00
static void copy_and_push_back(object* from, stack& to, unsigned int size)
{
2017-05-02 10:47:03 +02:00
// copy a whole stack entry and push it back to another stack
object* allocated = to.allocate_back(size, from->_type);
memcpy(allocated, from, size);
2017-05-02 10:47:03 +02:00
if (allocated->_type == cmd_number)
2017-06-14 22:17:49 +02:00
((number*)allocated)->move();
else if (allocated->_type == cmd_complex)
((complex*)allocated)->move();
2015-05-19 17:51:03 +02:00
}
2017-05-02 10:47:03 +02:00
object* allocate_back(unsigned int size, cmd_type_t type)
{
2017-05-02 10:47:03 +02:00
object* allocated;
2017-05-22 23:32:39 +02:00
bool data_is_reallocated = false;
char* old_base;
2017-05-22 23:32:39 +02:00
// manage data memory allocation (add as much as memory it is needed)
if (((_current - _base) + size) > _total_size)
{
// calc nb of needed pages
2017-05-22 23:32:39 +02:00
unsigned long page_number = 1 + ((_current - _base) + size - _total_size) / ALLOC_STACK_CHUNK;
_total_size += page_number * ALLOC_STACK_CHUNK;
old_base = _base;
_base = (char*)realloc(_base, _total_size);
2017-05-24 11:55:40 +02:00
_current = _base + (_current - old_base);
2017-06-10 11:43:10 +02:00
data_is_reallocated = true;
2017-05-22 18:22:15 +02:00
}
2017-05-22 23:32:39 +02:00
2017-05-24 11:55:40 +02:00
// manage pointers memory allocation (add one page if needed)
2017-05-22 23:32:39 +02:00
if ((_count + 1) > _total_count_pointer)
2017-05-22 18:22:15 +02:00
{
_base_pointer = (object**)realloc(_base_pointer, (_total_count_pointer * sizeof(object*)) + ALLOC_STACK_CHUNK);
_total_count_pointer += (ALLOC_STACK_CHUNK / sizeof(object));
2017-05-22 23:32:39 +02:00
}
2017-05-24 11:55:40 +02:00
// recalc object pointers in case of base reallocation
2017-05-22 23:32:39 +02:00
if (data_is_reallocated)
for(int i = 0; i < _count; i++)
_base_pointer[i] = (object*)(_base + ((char*)_base_pointer[i] - old_base));
2017-05-02 10:47:03 +02:00
// manage stack itself
2017-05-22 23:32:39 +02:00
_base_pointer[_count++] = (object*)_current;
2017-05-02 10:47:03 +02:00
allocated = (object*)_current;
_current += size;
2017-05-02 10:47:03 +02:00
// init object
allocated->_type = type;
allocated->_size = size;
if (type == cmd_number)
2017-06-14 22:17:49 +02:00
((number*)allocated)->init();
else if (type == cmd_complex)
((complex*)allocated)->init();
2017-05-02 10:47:03 +02:00
return allocated;
}
2017-05-02 10:47:03 +02:00
object* pop_back()
2015-05-19 17:51:03 +02:00
{
2017-05-02 10:47:03 +02:00
object* back = NULL;
2015-05-19 17:51:03 +02:00
if (_count > 0)
{
2017-05-24 11:55:40 +02:00
_current = (char*)_base_pointer[--_count];
2017-05-02 10:47:03 +02:00
back = (object*)_current;
2015-05-19 17:51:03 +02:00
}
return back;
2015-05-19 17:51:03 +02:00
}
unsigned int size()
{
return _count;
}
// stack access (stack_level=0=first out)
object* get_obj(unsigned int stack_level)
2015-05-19 17:51:03 +02:00
{
return seq_obj(_count - stack_level - 1);
2015-05-19 17:51:03 +02:00
}
object* operator[](unsigned int stack_level)
2015-05-19 17:51:03 +02:00
{
return seq_obj(_count - stack_level - 1);
2015-05-19 17:51:03 +02:00
}
2017-05-24 11:55:40 +02:00
2017-05-02 10:47:03 +02:00
object* back()
2015-05-19 17:51:03 +02:00
{
2017-05-24 11:55:40 +02:00
object* obj = NULL;
2015-05-19 17:51:03 +02:00
if (_count>0)
2017-05-24 11:55:40 +02:00
obj = _base_pointer[_count - 1];
return obj;
2015-05-19 17:51:03 +02:00
}
unsigned int get_len(unsigned int index)
{
2017-05-24 11:55:40 +02:00
return seq_len(_count - index - 1);
2015-05-19 17:51:03 +02:00
}
2017-05-02 10:47:03 +02:00
cmd_type_t get_type(unsigned int index)
2015-05-19 17:51:03 +02:00
{
2017-05-24 11:55:40 +02:00
return seq_type(_count - index - 1);
2015-05-19 17:51:03 +02:00
}
// sequential access (index is counted from front)
2017-05-02 10:47:03 +02:00
object* seq_obj(unsigned int index)
2015-05-19 17:51:03 +02:00
{
2017-05-24 11:55:40 +02:00
object* obj = NULL;
2015-05-19 17:51:03 +02:00
if (index<_count)
2017-05-24 11:55:40 +02:00
obj = _base_pointer[index];
return obj;
2015-05-19 17:51:03 +02:00
}
unsigned int seq_len(unsigned int index)
{
2017-05-24 11:55:40 +02:00
unsigned int len = 0;
2015-05-19 17:51:03 +02:00
if (index<_count)
2017-05-24 11:55:40 +02:00
len = _base_pointer[index]->_size;
return len;
2015-05-19 17:51:03 +02:00
}
2017-05-02 10:47:03 +02:00
cmd_type_t seq_type(unsigned int index)
2015-05-19 17:51:03 +02:00
{
2017-05-24 11:55:40 +02:00
cmd_type_t type = cmd_undef;
2015-05-19 17:51:03 +02:00
if (index<_count)
2017-05-24 11:55:40 +02:00
type = _base_pointer[index]->_type;
return type;
2015-05-19 17:51:03 +02:00
}
private:
2015-05-19 17:51:03 +02:00
char* _base;
char* _current;
2017-05-22 18:22:15 +02:00
object** _base_pointer;
2017-05-24 11:55:40 +02:00
unsigned int _count;//stack count
2017-05-22 18:22:15 +02:00
unsigned int _total_count_pointer;//total number of possible pointers
2017-05-24 11:55:40 +02:00
unsigned int _total_size;//total allocated data size in bytes
};
//
2017-05-22 18:22:15 +02:00
class heap : public stack
{
public:
2015-05-19 17:51:03 +02:00
heap() { }
2017-05-22 18:22:15 +02:00
virtual ~heap() { }
2015-05-19 17:51:03 +02:00
2017-05-22 18:22:15 +02:00
object* add(const string name, object* obj, unsigned int size)
2017-05-24 23:32:25 +02:00
{
map<string, unsigned int>::iterator i = _map.find(name);
object* local = NULL;
2017-05-22 18:22:15 +02:00
// variable does not exist in heap or already exists but its size is too short -> allocate
2017-05-24 23:32:25 +02:00
if (i!=_map.end())
local = seq_obj(i->second);
2017-05-22 18:22:15 +02:00
if (local==NULL || (local!=NULL && size>local->_size))
2015-05-19 17:51:03 +02:00
{
2017-05-22 18:22:15 +02:00
copy_and_push_back(obj, *this, size);
_map[name] = ((stack*)this)->size()-1;
2015-03-01 16:30:13 +01:00
}
2017-05-22 18:22:15 +02:00
else
2015-05-19 17:51:03 +02:00
{
2017-05-22 18:22:15 +02:00
// variable already exists in heap but previous was larger -> don't reallocate
// copy a whole stack entry and push it back to another stack
memcpy(local, obj, size);
if (local->_type == cmd_number)
2017-06-14 22:17:49 +02:00
((number*)local)->move();
else if (local->_type == cmd_complex)
((complex*)local)->move();
2015-03-01 16:30:13 +01:00
}
2017-05-17 18:15:03 +02:00
2017-05-22 18:22:15 +02:00
return local;
}
2017-05-02 10:47:03 +02:00
bool get(const string name, object*& obj, unsigned int& size)
2015-05-19 17:51:03 +02:00
{
2017-05-22 18:22:15 +02:00
bool ret = false;
2017-05-24 23:32:25 +02:00
map<string, unsigned int>::iterator i = _map.find(name);
2017-05-22 18:22:15 +02:00
2017-05-24 23:32:25 +02:00
if (i!=_map.end())
2015-05-19 17:51:03 +02:00
{
2017-05-24 23:32:25 +02:00
obj = seq_obj(i->second);
size = obj->_size;
2017-05-22 18:22:15 +02:00
ret = true;
2015-05-19 17:51:03 +02:00
}
2017-05-22 18:22:15 +02:00
return ret;
2015-05-19 17:51:03 +02:00
}
2017-05-17 18:15:03 +02:00
bool replace_value(const string name, object* obj, unsigned int size)
{
bool ret=false;
2017-05-24 23:32:25 +02:00
map<string, unsigned int>::iterator i = _map.find(name);
2017-05-17 18:15:03 +02:00
2017-05-24 23:32:25 +02:00
if (i!=_map.end())
2017-05-17 18:15:03 +02:00
{
2017-05-24 23:32:25 +02:00
object* obj_dst = seq_obj(i->second);
if (size<=obj_dst->_size)
{
(void)memcpy(obj_dst, obj, size);
if (obj_dst->_type == cmd_number)
2017-06-14 22:17:49 +02:00
((number*)obj_dst)->move();
else if (obj_dst->_type == cmd_complex)
((complex*)obj_dst)->move();
2017-05-24 23:32:25 +02:00
ret = true;
}
2017-05-17 18:15:03 +02:00
}
}
2015-05-19 17:51:03 +02:00
bool exist(const string name)
{
return (_map.find(name) != _map.end());
}
2017-05-02 10:47:03 +02:00
bool get_by_index(int num, string& name, object*& obj, unsigned int& size)
2015-05-19 17:51:03 +02:00
{
if (num>=0 && num<(int)_map.size())
{
2017-05-22 18:22:15 +02:00
object* local;
2017-05-24 23:32:25 +02:00
map<string, unsigned int>::iterator i= _map.begin();
2015-05-19 17:51:03 +02:00
2017-05-24 11:55:40 +02:00
for(int j = 0; j < num; j++)
2015-05-19 17:51:03 +02:00
i++;
2017-05-24 23:32:25 +02:00
local = (object*)seq_obj(i->second);
2015-05-19 17:51:03 +02:00
name = i->first;
2017-05-22 18:22:15 +02:00
obj = local;
size = local->_size;
2015-05-19 17:51:03 +02:00
return true;
}
else
return false;
}
2015-03-01 16:30:13 +01:00
bool erase(const string& name)
2015-05-19 17:51:03 +02:00
{
2017-05-24 23:32:25 +02:00
map<string, unsigned int>::iterator i = _map.find(name);
2017-05-22 18:22:15 +02:00
bool ret = false;
2015-05-19 17:51:03 +02:00
if (i != _map.end())
{
// remove variable from map
2015-05-19 17:51:03 +02:00
_map.erase(i->first);
2017-05-22 18:22:15 +02:00
ret = true;
// TODO: remove unused stack entries
2015-05-19 17:51:03 +02:00
}
2017-05-22 18:22:15 +02:00
return ret;
2015-05-19 17:51:03 +02:00
}
2017-05-31 22:39:14 +02:00
void erase_all(void)
{
// map
_map.erase(_map.begin(), _map.end());
// and stack
((stack*)this)->erase();
}
unsigned int count_vars() { return _map.size(); }
private:
2017-05-24 23:32:25 +02:00
map<string, unsigned int> _map;
};
#endif // __stack_h__