Program Listing for File AgentVector.cpp
↰ Return to documentation for file (src/flamegpu/simulation/AgentVector.cpp
)
#include "flamegpu/simulation/AgentVector.h"
#include <limits>
#include <utility>
#include <string>
#include <memory>
#include "flamegpu/model/AgentDescription.h"
#include "flamegpu/simulation/AgentVector_Agent.h"
// @todo - this shouldn't be required anymore?
#ifdef max
#undef max // Unclear where this definition is leaking from
#endif
namespace flamegpu {
const float AgentVector::RESIZE_FACTOR = 1.5f;
AgentVector::AgentVector(const CAgentDescription& agent_desc, size_type count)
: AgentVector(*agent_desc.agent, count) { }
AgentVector::AgentVector(const AgentData& agent_desc, size_type count)
: agent(agent_desc.clone())
, _size(0)
, _capacity(0)
, _data(std::make_shared<AgentDataMap>()) {
resize(count);
}
AgentVector::AgentVector(const AgentVector& other)
: agent(other.agent->clone())
, _size(0)
, _capacity(0)
, _data(std::make_shared<AgentDataMap>()) {
clear();
insert(0, other.begin(), other.end());
}
AgentVector::AgentVector(AgentVector&& other) noexcept
: agent(other.agent->clone())
, _size(other._size)
, _capacity(other._capacity)
, _data(std::make_shared<AgentDataMap>()) {
// Purge our data
_data->clear();
// Swap data
std::swap(*_data, *other._data); // Not 100% sure this will work as intended
// Purge other
other._size = 0;
other._capacity = 0;
}
AgentVector& AgentVector::operator=(const AgentVector& other) {
// Self assignment
if (this == &other)
return *this;
if (*agent != *other.agent) {
throw std::exception(); // AgentVectors are for different AgentDescriptions
}
// Copy size
internal_resize(other.size(), false);
_size = other.size();
if (_size) {
// Copy data
for (const auto& v : agent->variables) {
auto &this_it = _data->at(v.first);
const auto &other_it = other._data->at(v.first);
memcpy(this_it->getDataPtr(), other_it->getReadOnlyDataPtr(), _size * v.second.type_size * v.second.elements);
}
}
return *this;
}
AgentVector& AgentVector::operator=(AgentVector&& other) noexcept {
agent = other.agent->clone();
_size = other._size;
_capacity = other._capacity;
// Purge our data
_data->clear();
// Swap data
std::swap(*_data, *other._data); // Not 100% sure this will work as intended
// Purge other
other._size = 0;
other._capacity = 0;
return *this;
}
AgentVector::Agent AgentVector::at(size_type pos) {
if (pos >= _size) {
_requireLength();
if (pos >= _size) {
THROW exception::OutOfBoundsException("pos (%u) exceeds length of vector (%u) in AgentVector::at()", pos, _size);
}
}
// Return the agent instance
return Agent(this, agent, _data, pos);
}
AgentVector::CAgent AgentVector::at(size_type pos) const {
if (pos >= _size) {
_requireLength();
if (pos >= _size) {
THROW exception::OutOfBoundsException("pos (%u) exceeds length of vector (%u) in AgentVector::at()", pos, _size);
}
}
return CAgent(const_cast<AgentVector*>(this), agent, _data, pos);
}
AgentVector::Agent AgentVector::operator[](size_type pos) {
return at(pos);
}
AgentVector::CAgent AgentVector::operator[](size_type pos) const {
return at(pos);
}
AgentVector::Agent AgentVector::front() {
return at(0);
}
AgentVector::CAgent AgentVector::front() const {
return at(0);
}
AgentVector::Agent AgentVector::back() {
_requireLength();
return at(_size - 1);
}
AgentVector::CAgent AgentVector::back() const {
_requireLength();
return at(_size - 1);
}
void* AgentVector::data(const std::string& variable_name) {
if (!variable_name.empty() && variable_name[0] == '_') {
THROW exception::ReservedName("Agent variable names that begin with '_' are reserved for internal usage and cannot be changed directly, "
"in AgentVector::data().");
}
// Is variable name found
const auto& var = agent->variables.find(variable_name);
if (var == agent->variables.end()) {
THROW exception::InvalidAgentVar("Variable with name '%s' was not found in agent '%s', "
"in AgentVector::data().",
variable_name.c_str(), agent->name.c_str());
}
// Does the map have a vector
const auto& map_it = _data->find(variable_name);
if (map_it != _data->end()) {
_requireLength();
_require(variable_name);
_changedAfter(variable_name, 0);
return map_it->second->getDataPtr();
}
return nullptr;
}
const void* AgentVector::data(const std::string& variable_name) const {
// Is variable name found
const auto& var = agent->variables.find(variable_name);
if (var == agent->variables.end()) {
THROW exception::InvalidAgentVar("Variable with name '%s' was not found in agent '%s', "
"in AgentVector::data().",
variable_name.c_str(), agent->name.c_str());
}
// Does the map have a vector
const auto& map_it = _data->find(variable_name);
if (map_it != _data->end()) {
_requireLength();
_require(variable_name);
return map_it->second->getDataPtr();
}
return nullptr;
}
AgentVector::iterator AgentVector::begin() noexcept {
_requireLength();
return iterator(this, agent, _data, 0);
}
AgentVector::const_iterator AgentVector::begin() const noexcept {
_requireLength();
return const_iterator(const_cast<AgentVector*>(this), agent, _data, 0);
}
AgentVector::const_iterator AgentVector::cbegin() const noexcept {
_requireLength();
return const_iterator(const_cast<AgentVector*>(this), agent, _data, 0);
}
AgentVector::iterator AgentVector::end() noexcept {
return iterator(this, agent, _data, _size);
}
AgentVector::const_iterator AgentVector::end() const noexcept {
return const_iterator(const_cast<AgentVector*>(this), agent, _data, _size);
}
AgentVector::const_iterator AgentVector::cend() const noexcept {
return const_iterator(const_cast<AgentVector*>(this), agent, _data, _size);
}
AgentVector::reverse_iterator AgentVector::rbegin() noexcept {
_requireLength();
return reverse_iterator(this, agent, _data, _size - 1);
}
AgentVector::const_reverse_iterator AgentVector::rbegin() const noexcept {
_requireLength();
return const_reverse_iterator(const_cast<AgentVector*>(this), agent, _data, _size-1);
}
AgentVector::const_reverse_iterator AgentVector::crbegin() const noexcept {
_requireLength();
return const_reverse_iterator(const_cast<AgentVector*>(this), agent, _data, _size-1);
}
AgentVector::reverse_iterator AgentVector::rend() noexcept {
return reverse_iterator(this, agent, _data, std::numeric_limits<size_type>::max());
}
AgentVector::const_reverse_iterator AgentVector::rend() const noexcept {
return const_reverse_iterator(const_cast<AgentVector*>(this), agent, _data, std::numeric_limits<size_type>::max());
}
AgentVector::const_reverse_iterator AgentVector::crend() const noexcept {
return const_reverse_iterator(const_cast<AgentVector*>(this), agent, _data, std::numeric_limits<size_type>::max());
}
bool AgentVector::empty() const {
_requireLength();
return _size == 0;
}
flamegpu::size_type AgentVector::size() const {
_requireLength();
return _size;
}
flamegpu::size_type AgentVector::max_size() { return std::numeric_limits<size_type>::max() - 1; }
void AgentVector::reserve(size_type new_cap) {
if (new_cap > _capacity) {
internal_resize(new_cap, true);
}
}
flamegpu::size_type AgentVector::capacity() const { return _capacity; }
void AgentVector::shrink_to_fit() {
_requireLength();
if (_size > _capacity) {
internal_resize(_size, false);
}
}
void AgentVector::clear() {
_requireLength();
// Re initialise all variables
if (_capacity) {
init(0, _size);
}
_size = 0;
_erase(0, _size);
}
#ifdef _MSC_VER
#pragma warning(push, 1)
#pragma warning(disable : 4127)
// Suppress condition expression is constant
// The constant condition can be made constexpr in future with C++17
#endif
void AgentVector::resetAllIDs() {
_require(ID_VARIABLE_NAME);
const auto it = _data->find(ID_VARIABLE_NAME);
if (it != _data->end()) {
constexpr id_t DEFAULT_VALUE = ID_NOT_SET;
id_t* t_data = static_cast<id_t*>(it->second->getDataPtr());
if (DEFAULT_VALUE == 0) {
memset(t_data, 0, _size * sizeof(id_t));
} else {
for (unsigned int i = 0; i < _size; ++i) {
memcpy(t_data + i, &DEFAULT_VALUE, sizeof(id_t));
}
}
} else {
THROW exception::InvalidOperation("Agent '%s' is missing internal ID variable, "
"in AgentVector::resetAllIDs()\n",
agent->name.c_str());
}
if (_size) {
// Mark all indices as changed (there isn't currently a single fn for this)
_changed(ID_VARIABLE_NAME, 0);
_changedAfter(ID_VARIABLE_NAME, 0);
}
}
#ifdef _MSC_VER
#pragma warning(pop)
#endif
void AgentVector::init(size_type first, size_type last) {
if (first >= last) {
THROW exception::InvalidOperation("Last (%u) must exceed first(%u), "
"in AgentVector::init()\n",
last, first);
} else if (last > _capacity) {
THROW exception::OutOfBoundsException("Last (%u) exceeds capacity (%u) in AgentVector::init(), "
"in AgentVector::init()\n",
last, _capacity);
}
// Re initialise all variables
for (const auto& v : agent->variables) {
const auto it = _data->find(v.first);
const size_t variable_size = v.second.type_size * v.second.elements;
char* t_data = static_cast<char*>(it->second->getDataPtr());
for (unsigned int i = first; i < last; ++i) {
memcpy(t_data + i * variable_size, v.second.default_value, variable_size);
}
}
}
AgentVector::iterator AgentVector::insert(const_iterator pos, const AgentInstance& value) {
return insert(pos, 1, value);
}
AgentVector::iterator AgentVector::insert(size_type pos, const AgentInstance& value) {
return insert(pos, 1, value);
}
AgentVector::iterator AgentVector::insert(const_iterator pos, const Agent& value) {
return insert(pos, 1, value);
}
AgentVector::iterator AgentVector::insert(size_type pos, const Agent& value) {
return insert(pos, 1, value);
}
AgentVector::iterator AgentVector::insert(const_iterator pos, size_type count, const AgentInstance& value) {
return insert(pos._pos, count, value);
}
AgentVector::iterator AgentVector::insert(size_type pos, size_type count, const AgentInstance& value) {
if (count == 0)
return iterator(this, agent, _data, pos);
// Confirm they are for the same agent type
if (value._agent != agent && *value._agent != *agent) {
THROW exception::InvalidAgent("Agent description mismatch, '%' provided, '%' required, "
"in AgentVector::push_back().\n",
value._agent->name.c_str(), agent->name.c_str());
}
// Expand capacity if required
{
_requireLength();
size_type new_capacity = _capacity;
assert((_capacity * RESIZE_FACTOR) + 1 > _capacity);
while (_size + count > new_capacity) {
new_capacity = static_cast<size_type>(new_capacity * RESIZE_FACTOR) + 1;
}
internal_resize(new_capacity, true);
}
// If we are not appending, ensure we have upto date device data
if (pos < _size)
_requireAll();
// Get first index;
const size_type insert_index = pos;
// Fix each variable
for (const auto& v : agent->variables) {
const auto it = _data->find(v.first);
char* t_data = static_cast<char*>(it->second->getDataPtr());
const size_t variable_size = v.second.type_size * v.second.elements;
// Move all items after this index backwards count places
for (unsigned int i = _size - 1; i >= insert_index; --i) {
// Copy items individually, incase the src and destination overlap
memcpy(t_data + (i + count) * variable_size, t_data + i * variable_size, variable_size);
}
// Copy across item data
const auto other_it = value._data.find(v.first);
for (unsigned int i = insert_index; i < insert_index + count; ++i) {
memcpy(t_data + i * variable_size, other_it->second.ptr, variable_size);
}
}
// Increase size
_size += count;
// Notify subclasses
_insert(pos, count);
// Return iterator to first inserted item
return iterator(this, agent, _data, insert_index);
}
AgentVector::iterator AgentVector::insert(const_iterator pos, size_type count, const Agent& value) {
return insert(pos._pos, count, value);
}
AgentVector::iterator AgentVector::insert(size_type pos, size_type count, const Agent& value) {
if (count == 0)
return iterator(this, agent, _data, pos);
// Confirm they are for the same agent type
if (value._agent != agent && *value._agent != *agent) {
THROW exception::InvalidAgent("Agent description mismatch, '%' provided, '%' required, "
"in AgentVector::push_back().\n",
value._agent->name.c_str(), agent->name.c_str());
}
// Expand capacity if required
{
_requireLength();
size_type new_capacity = _capacity;
assert((_capacity * RESIZE_FACTOR) + 1 > _capacity);
while (_size + count > new_capacity) {
new_capacity = static_cast<size_type>(new_capacity * RESIZE_FACTOR) + 1;
}
internal_resize(new_capacity, true);
}
// If we are not appending, ensure we have upto date device data
if (pos < _size)
_requireAll();
// Get first index;
const size_type insert_index = pos;
// Fix each variable
auto value_data = value._data.lock();
if (!value_data) {
THROW exception::ExpiredWeakPtr("The AgentVector which owns the passed AgentVector::Agent has been deallocated, "
"in AgentVector::insert().\n");
}
const id_t ID_DEFAULT = ID_NOT_SET;
for (const auto& v : agent->variables) {
const auto it = _data->find(v.first);
char* t_data = static_cast<char*>(it->second->getDataPtr());
const size_t variable_size = v.second.type_size * v.second.elements;
// Move all items after this index backwards count places
for (unsigned int i = _size - 1; i >= insert_index; --i) {
// Copy items individually, incase the src and destination overlap
memcpy(t_data + (i + count) * variable_size, t_data + i * variable_size, variable_size);
}
// Copy across item data, ID has a special case, where it is default init instead of being copied
if (v.first == ID_VARIABLE_NAME) {
if (v.second.elements != 1 || v.second.type != std::type_index(typeid(id_t))) {
THROW exception::InvalidOperation("Agent's internal ID variable is not type %s[1], "
"in AgentVector::insert()\n", std::type_index(typeid(id_t)).name());
}
for (unsigned int i = insert_index; i < insert_index + count; ++i) {
memcpy(t_data + i * variable_size, &ID_DEFAULT, sizeof(id_t));
}
} else {
const auto other_it = value_data->find(v.first);
char* src_data = static_cast<char*>(other_it->second->getDataPtr());
for (unsigned int i = insert_index; i < insert_index + count; ++i) {
memcpy(t_data + i * variable_size, src_data + value.index * variable_size, variable_size);
}
}
}
// Increase size
_size += count;
// Notify subclasses
_insert(pos, count);
// Return iterator to first inserted item
return iterator(this, agent, _data, insert_index);
}
AgentVector::iterator AgentVector::erase(const_iterator pos) {
const auto first = pos++;
return erase(first._pos, pos._pos);
}
AgentVector::iterator AgentVector::erase(size_type pos) {
const auto first = pos++;
return erase(first, pos);
}
AgentVector::iterator AgentVector::erase(const_iterator first, const_iterator last) {
// Confirm they are for the same agent type
if (first._agent != agent && *first._agent != *agent) {
THROW exception::InvalidAgent("Agent description mismatch, '%' provided, '%' required, "
"in AgentVector::push_back().\n",
first._agent->name.c_str(), agent->name.c_str());
}
if (last._agent != agent && *last._agent != *agent) {
THROW exception::InvalidAgent("Agent description mismatch, '%' provided, '%' required, "
"in AgentVector::push_back().\n",
last._agent->name.c_str(), agent->name.c_str());
}
return erase(first._pos, last._pos);
}
AgentVector::iterator AgentVector::erase(size_type first, size_type last) {
if (first == last)
return iterator(this, agent, _data, last);
// Get first index;
const size_type first_remove_index = first < last ? first : last;
const size_type first_move_index = first < last ? last : first;
const size_type erase_count = first_move_index - first_remove_index;
_requireLength();
const size_type first_empty_index = _size - erase_count;
const size_type last_empty_index = _size;
// Ensure indicies are in bounds
if (first_remove_index >= _size) {
THROW exception::OutOfBoundsException("%u is not a valid index into the vector, "
"in AgentVector::erase()\n", first_remove_index);
} else if (first_move_index > _size) {
THROW exception::OutOfBoundsException("%u is not a valid index to the end of a range of vector items, "
"it must point to after the final selected item, "
"in AgentVector::erase()\n", first_move_index);
}
// If we are not erasing from the end, ensure we have upto date device data
if (first_move_index != _size)
_requireAll();
// Fix each variable
for (const auto& v : agent->variables) {
const auto it = _data->find(v.first);
char* t_data = static_cast<char*>(it->second->getDataPtr());
const size_t variable_size = v.second.type_size * v.second.elements;
// Move all items after this index forwards count places
for (unsigned int i = first_move_index; i < _size; ++i) {
// Copy items individually, incase the src and destination overlap
memcpy(t_data + (i - erase_count) * variable_size, t_data + i * variable_size, variable_size);
}
}
// Initialise newly empty variables
init(first_empty_index, last_empty_index);
// Decrease size
_size -= erase_count;
// Notify subclasses
_erase(first_remove_index, erase_count);
// Return iterator following the last removed element
return iterator(this, agent, _data, first_remove_index + 1);
}
void AgentVector::push_back(const AgentInstance& value) {
insert(cend(), value);
}
void AgentVector::push_back() {
// Expand capacity if required
{
_requireLength();
size_type new_capacity = _capacity;
assert((_capacity * RESIZE_FACTOR) + 1 > _capacity);
while (_size + 1 > new_capacity) {
new_capacity = static_cast<size_type>(_capacity * RESIZE_FACTOR) + 1;
}
internal_resize(new_capacity, true);
}
// Notify subclasses & increase size
_insert(_size++, 1);
}
void AgentVector::pop_back() {
_requireLength();
if (_size) {
--_size;
// Reset removed item to default value
for (const auto& v : agent->variables) {
const auto it = _data->find(v.first);
const size_t variable_size = v.second.type_size * v.second.elements;
char* t_data = static_cast<char*>(it->second->getDataPtr());
memcpy(t_data + _size * variable_size, v.second.default_value, variable_size);
}
// Notify subclasses
_erase(_size, 1);
}
}
void AgentVector::resize(size_type count) {
_requireLength();
const size_type old_size = _size;
internal_resize(count, true);
_size = count;
// Notify subclasses
if (count > old_size) {
_insert(old_size, count - old_size);
} else if (count < old_size) {
_erase(count, old_size - count);
}
}
void AgentVector::internal_resize(size_type count, bool init) {
if (count == _capacity)
return;
for (const auto& v : agent->variables) {
// For each variable inside agent, add it to the map or replace it in the map
const auto it = _data->find(v.first);
const size_t variable_size = v.second.type_size * v.second.elements;
if (it == _data->end()) {
// Need to create the variable's vector
auto t = std::unique_ptr<detail::GenericMemoryVector>(v.second.memory_vector->clone());
t->resize(count);
// Default init all new elements
if (init) {
char* t_data = static_cast<char*>(t->getDataPtr());
for (unsigned int i = 0; i < count; ++i) {
memcpy(t_data + i * variable_size, v.second.default_value, variable_size);
}
}
_data->emplace(v.first, std::move(t));
} else {
// Need to resize the variables vector
it->second->resize(count);
}
}
size_type old_capacity = _capacity;
_capacity = count;
// Default init all new elements
if (init && count > old_capacity) {
this->init(old_capacity, _capacity);
}
}
void AgentVector::swap(AgentVector& other) noexcept {
std::swap(_data, other._data);
std::swap(_capacity, other._capacity);
std::swap(_size, other._size);
std::swap(agent, other.agent);
}
bool AgentVector::operator==(const AgentVector& other) const {
_requireLength();
if (_size != other._size)
return false;
if (*agent != *other.agent)
return false;
_requireAll();
for (const auto& v : agent->variables) {
const auto it_a = _data->find(v.first);
const auto it_b = other._data->find(v.first);
const char* data_a = static_cast<const char*>(it_a->second->getReadOnlyDataPtr());
const char* data_b = static_cast<const char*>(it_b->second->getReadOnlyDataPtr());
for (size_type i = 0; i < _size; ++i)
if (data_a[i] != data_b[i])
return false;
}
return true;
}
bool AgentVector::operator!=(const AgentVector& other) const {
return !((*this) == other);
}
bool AgentVector::matchesAgentType(const AgentData& other) const { return *agent == other; }
bool AgentVector::matchesAgentType(const CAgentDescription& other) const { return *agent == *other.agent; }
std::type_index AgentVector::getVariableType(const std::string &variable_name) const {
const auto &it = agent->variables.find(variable_name);
if (it == agent->variables.end()) {
THROW exception::InvalidAgentVar("Agent '%s' does not contain variable with name '%s', "
"in AgentVector::getVariableType()\n",
agent->name.c_str(), variable_name.c_str());
}
return it->second.type;
}
const VariableMap& AgentVector::getVariableMetaData() const {
return agent->variables;
}
std::string AgentVector::getInitialState() const {
return agent->initial_state;
}
AgentVector::Agent AgentVector::iterator::operator*() const {
return Agent(_parent, _agent, _data, _pos);
}
AgentVector::CAgent AgentVector::const_iterator::operator*() const {
return CAgent(_parent, _agent, _data, _pos);
}
AgentVector::Agent AgentVector::reverse_iterator::operator*() const {
return Agent(_parent, _agent, _data, _pos);
}
AgentVector::CAgent AgentVector::const_reverse_iterator::operator*() const {
return CAgent(_parent, _agent, _data, _pos);
}
} // namespace flamegpu