BizHawk/waterbox/ares64/ares/nall/string/allocator/adaptive.hpp

124 lines
2.9 KiB
C++

#pragma once
/*****
adaptive allocator
sizeof(string) == SSO + 8
aggressively tries to avoid heap allocations
small strings are stored on the stack
large strings are shared via copy-on-write
SSO alone is very slow on large strings due to copying
SSO alone is very slightly faster than this allocator on small strings
COW alone is very slow on small strings due to heap allocations
COW alone is very slightly faster than this allocator on large strings
adaptive is thus very fast for all string sizes
*****/
namespace nall {
inline string::string() : _data(nullptr), _capacity(SSO - 1), _size(0) {
}
template<typename T>
inline auto string::get() -> T* {
if(_capacity < SSO) return (T*)_text;
if(*_refs > 1) _copy();
return (T*)_data;
}
template<typename T>
inline auto string::data() const -> const T* {
if(_capacity < SSO) return (const T*)_text;
return (const T*)_data;
}
inline auto string::reset() -> type& {
if(_capacity >= SSO && !--*_refs) memory::free(_data);
_data = nullptr;
_capacity = SSO - 1;
_size = 0;
return *this;
}
inline auto string::reserve(u32 capacity) -> type& {
if(capacity <= _capacity) return *this;
capacity = bit::round(capacity + 1) - 1;
if(_capacity < SSO) {
_capacity = capacity;
_allocate();
} else if(*_refs > 1) {
_capacity = capacity;
_copy();
} else {
_capacity = capacity;
_resize();
}
return *this;
}
inline auto string::resize(u32 size) -> type& {
reserve(size);
get()[_size = size] = 0;
return *this;
}
inline auto string::operator=(const string& source) -> type& {
if(&source == this) return *this;
reset();
if(source._capacity >= SSO) {
_data = source._data;
_refs = source._refs;
_capacity = source._capacity;
_size = source._size;
++*_refs;
} else {
memory::copy(_text, source._text, SSO);
_capacity = source._capacity;
_size = source._size;
}
return *this;
}
inline auto string::operator=(string&& source) -> type& {
if(&source == this) return *this;
reset();
memory::copy(this, &source, sizeof(string));
source._data = nullptr;
source._capacity = SSO - 1;
source._size = 0;
return *this;
}
//SSO -> COW
inline auto string::_allocate() -> void {
char _temp[SSO];
memory::copy(_temp, _text, SSO);
_data = memory::allocate<char>(_capacity + 1 + sizeof(u32));
memory::copy(_data, _temp, SSO);
_refs = (u32*)(_data + _capacity + 1); //always aligned by 32 via reserve()
*_refs = 1;
}
//COW -> Unique
inline auto string::_copy() -> void {
auto _temp = memory::allocate<char>(_capacity + 1 + sizeof(u32));
memory::copy(_temp, _data, _size = min(_capacity, _size));
_temp[_size] = 0;
--*_refs;
_data = _temp;
_refs = (u32*)(_data + _capacity + 1);
*_refs = 1;
}
//COW -> Resize
inline auto string::_resize() -> void {
_data = memory::resize<char>(_data, _capacity + 1 + sizeof(u32));
_refs = (u32*)(_data + _capacity + 1);
*_refs = 1;
}
}