.. _program_listing_file_include_flamegpu_runtime_environment_DeviceEnvironmentDirectedGraph.cuh: Program Listing for File DeviceEnvironmentDirectedGraph.cuh =========================================================== |exhale_lsh| :ref:`Return to documentation for file ` (``include/flamegpu/runtime/environment/DeviceEnvironmentDirectedGraph.cuh``) .. |exhale_lsh| unicode:: U+021B0 .. UPWARDS ARROW WITH TIP LEFTWARDS .. code-block:: cpp #ifndef INCLUDE_FLAMEGPU_RUNTIME_ENVIRONMENT_DEVICEENVIRONMENTDIRECTEDGRAPH_CUH_ #define INCLUDE_FLAMEGPU_RUNTIME_ENVIRONMENT_DEVICEENVIRONMENTDIRECTEDGRAPH_CUH_ #include "flamegpu/defines.h" #include "flamegpu/runtime/detail/curve/Curve.cuh" namespace flamegpu { class DeviceEnvironmentDirectedGraph { const detail::curve::Curve::VariableHash graph_hash; public: class OutEdgeFilter { friend class Message; public: class Edge { const OutEdgeFilter&_parent; unsigned int edge_index; public: __device__ Edge(const OutEdgeFilter&parent, const unsigned int&_edge_index) : _parent(parent) , edge_index(_edge_index) { } __device__ bool operator==(const Edge& rhs) const { return this->edge_index == rhs.edge_index; } __device__ bool operator!=(const Edge& rhs) const { return !(*this == rhs); } __device__ Edge& operator++() { ++edge_index; return *this; } __device__ unsigned int getIndex() const { return edge_index; } template __device__ T getProperty(const char(&property_name)[N]) const; template __device__ T getProperty(const char(&property_name)[M], const unsigned int &index) const; __device__ unsigned int getEdgeDestination() const; }; class iterator { Edge _edge; public: __device__ iterator(const OutEdgeFilter &parent, const unsigned int &cell_index) : _edge(parent, cell_index) { // Increment to find first edge ++_edge; } __device__ iterator& operator++() { ++_edge; return *this; } __device__ iterator operator++(int) { iterator temp = *this; ++*this; return temp; } __device__ bool operator==(const iterator& rhs) const { return _edge == rhs._edge; } __device__ bool operator!=(const iterator& rhs) const { return _edge != rhs._edge; } __device__ Edge& operator*() { return _edge; } __device__ Edge* operator->() { return &_edge; } }; inline __device__ OutEdgeFilter(detail::curve::Curve::VariableHash _graph_hash, const id_t &vertex_index); inline __device__ iterator begin(void) const { // Bin before initial bin, as the constructor calls increment operator return iterator(*this, bucket_begin - 1); } inline __device__ iterator end(void) const { // Final bin, as the constructor calls increment operator return iterator(*this, bucket_end - 1); } inline __device__ unsigned int size(void) const { return bucket_end - bucket_begin; } private: unsigned int bucket_begin, bucket_end; const detail::curve::Curve::VariableHash graph_hash; }; class InEdgeFilter { friend class Message; public: class Edge { const InEdgeFilter&_parent; unsigned int ipbm_index; unsigned int edge_index; public: __device__ Edge(const InEdgeFilter&parent, const unsigned int&_ipbm_index) : _parent(parent) , ipbm_index(_ipbm_index) //, edge_index(0) // Value doesn't matter until operator++() is first called { } __device__ bool operator==(const Edge& rhs) const { return this->ipbm_index == rhs.ipbm_index; } __device__ bool operator!=(const Edge& rhs) const { return !(*this == rhs); } __device__ Edge& operator++() { // @todo This may not exit safely it SEATBELTS has left graph_ipbm_edges=nullptr? edge_index = _parent.graph_ipbm_edges[++ipbm_index]; return *this; } __device__ unsigned int getIndex() const { return edge_index; } template __device__ T getProperty(const char(&property_name)[N]) const; template __device__ T getProperty(const char(&property_name)[M], const unsigned int &index) const; __device__ unsigned int getEdgeSource() const; }; class iterator { Edge _edge; public: __device__ iterator(const InEdgeFilter &parent, const unsigned int &cell_index) : _edge(parent, cell_index) { // Increment to find first edge ++_edge; } __device__ iterator& operator++() { ++_edge; return *this; } __device__ iterator operator++(int) { iterator temp = *this; ++*this; return temp; } __device__ bool operator==(const iterator& rhs) const { return _edge == rhs._edge; } __device__ bool operator!=(const iterator& rhs) const { return _edge != rhs._edge; } __device__ Edge& operator*() { return _edge; } __device__ Edge* operator->() { return &_edge; } }; inline __device__ InEdgeFilter(detail::curve::Curve::VariableHash _graph_hash, const id_t &vertex_index); inline __device__ iterator begin(void) const { // Bin before initial bin, as the constructor calls increment operator return iterator(*this, bucket_begin - 1); } inline __device__ iterator end(void) const { // Final bin, as the constructor calls increment operator return iterator(*this, bucket_end - 1); } inline __device__ unsigned int size(void) const { return bucket_end - bucket_begin; } private: unsigned int bucket_begin, bucket_end; unsigned int *graph_ipbm_edges; const detail::curve::Curve::VariableHash graph_hash; }; __device__ __forceinline__ DeviceEnvironmentDirectedGraph(const detail::curve::Curve::VariableHash _graph_hash) : graph_hash(_graph_hash) { } __device__ __forceinline__ id_t getVertexID(unsigned int vertex_index) const; __device__ __forceinline__ unsigned int getVertexIndex(id_t vertex_id) const; template __device__ __forceinline__ T getVertexProperty(const char(&property_name)[M], unsigned int vertex_index) const; template __device__ __forceinline__ T getVertexProperty(const char(&property_name)[M], unsigned int vertex_index, unsigned int element_index) const; __device__ __forceinline__ id_t getEdgeSource(unsigned int edge_index) const; __device__ __forceinline__ id_t getEdgeDestination(unsigned int edge_index) const; __device__ __forceinline__ unsigned int getEdgeIndex(unsigned int source_vertex_index, unsigned int destination_vertex_index) const; template __device__ __forceinline__ T getEdgeProperty(const char(&property_name)[M], unsigned int edge_index) const; template __device__ __forceinline__ T getEdgeProperty(const char(&property_name)[M], unsigned int edge_index, unsigned int element_index) const; inline __device__ OutEdgeFilter outEdges(const id_t & vertex_index) const { return OutEdgeFilter(graph_hash, vertex_index); } inline __device__ InEdgeFilter inEdges(const id_t& vertex_index) const { return InEdgeFilter(graph_hash, vertex_index); } }; __device__ DeviceEnvironmentDirectedGraph::OutEdgeFilter::OutEdgeFilter(const detail::curve::Curve::VariableHash _graph_hash, const id_t& vertex_index) : bucket_begin(0) , bucket_end(0) , graph_hash(_graph_hash) { #if !defined(FLAMEGPU_SEATBELTS) || FLAMEGPU_SEATBELTS // Vertex "_id" always exists const unsigned int VERTEX_COUNT = detail::curve::DeviceCurve::getVariableCount("_id", graph_hash ^ detail::curve::Curve::variableHash("_environment_directed_graph_vertex")); if (vertex_index >= VERTEX_COUNT) { DTHROW("Vertex index (%u) exceeds vertex count (%u), unable to iterate outgoing edges.\n", vertex_index, VERTEX_COUNT); return; } #endif unsigned int* pbm = detail::curve::DeviceCurve::getEnvironmentDirectedGraphPBM(graph_hash); #if !defined(FLAMEGPU_SEATBELTS) || FLAMEGPU_SEATBELTS if (!pbm) return; #endif bucket_begin = pbm[vertex_index]; bucket_end = pbm[vertex_index + 1]; } __device__ DeviceEnvironmentDirectedGraph::InEdgeFilter::InEdgeFilter(const detail::curve::Curve::VariableHash _graph_hash, const id_t& vertex_index) : bucket_begin(0) , bucket_end(0) , graph_ipbm_edges(nullptr) , graph_hash(_graph_hash) { #if !defined(FLAMEGPU_SEATBELTS) || FLAMEGPU_SEATBELTS // Vertex "_id" always exists const unsigned int VERTEX_COUNT = detail::curve::DeviceCurve::getVariableCount("_id", graph_hash ^ detail::curve::Curve::variableHash("_environment_directed_graph_vertex")); if (vertex_index >= VERTEX_COUNT) { DTHROW("Vertex index (%u) exceeds vertex count (%u), unable to iterate incoming edges.\n", vertex_index, VERTEX_COUNT); return; } #endif unsigned int* ipbm = detail::curve::DeviceCurve::getEnvironmentDirectedGraphIPBM(graph_hash); #if !defined(FLAMEGPU_SEATBELTS) || FLAMEGPU_SEATBELTS if (!ipbm) return; #endif bucket_begin = ipbm[vertex_index]; bucket_end = ipbm[vertex_index + 1]; // Grab and store a copy of the PBM edgelist pointer this->graph_ipbm_edges = detail::curve::DeviceCurve::getEnvironmentDirectedGraphIPBMEdges(_graph_hash); } template __device__ T DeviceEnvironmentDirectedGraph::OutEdgeFilter::Edge::getProperty(const char(&property_name)[N]) const { #if !defined(FLAMEGPU_SEATBELTS) || FLAMEGPU_SEATBELTS if (edge_index >= _parent.bucket_end) { DTHROW("Edge index exceeds bin length, unable to get property '%s'.\n", property_name); return {}; } #endif // get the value from curve using the message index. T value = detail::curve::DeviceCurve::getEnvironmentDirectedGraphEdgeProperty(_parent.graph_hash, property_name, edge_index); return value; } template __device__ T DeviceEnvironmentDirectedGraph::OutEdgeFilter::Edge::getProperty(const char(&property_name)[M], const unsigned int& element_index) const { #if !defined(FLAMEGPU_SEATBELTS) || FLAMEGPU_SEATBELTS if (edge_index >= _parent.bucket_end) { DTHROW("Edge index exceeds bin length, unable to get property '%s'.\n", property_name); return {}; } #endif // get the value from curve using the message index. T value = detail::curve::DeviceCurve::getEnvironmentDirectedGraphEdgeArrayProperty(_parent.graph_hash, property_name, edge_index, element_index); return value; } template __device__ T DeviceEnvironmentDirectedGraph::InEdgeFilter::Edge::getProperty(const char(&property_name)[N]) const { #if !defined(FLAMEGPU_SEATBELTS) || FLAMEGPU_SEATBELTS const unsigned int EDGE_COUNT = detail::curve::DeviceCurve::getVariableCount("_source_dest", _parent.graph_hash ^ detail::curve::Curve::variableHash("_environment_directed_graph_edge")); if (edge_index >= EDGE_COUNT) { DTHROW("Edge index exceeds edge count, unable to get property '%s'.\n", property_name); return {}; } #endif // get the value from curve using the message index. T value = detail::curve::DeviceCurve::getEnvironmentDirectedGraphEdgeProperty(_parent.graph_hash, property_name, edge_index); return value; } template __device__ T DeviceEnvironmentDirectedGraph::InEdgeFilter::Edge::getProperty(const char(&property_name)[M], const unsigned int& element_index) const { #if !defined(FLAMEGPU_SEATBELTS) || FLAMEGPU_SEATBELTS const unsigned int EDGE_COUNT = detail::curve::DeviceCurve::getVariableCount("_source_dest", _parent.graph_hash ^ detail::curve::Curve::variableHash("_environment_directed_graph_edge")); if (edge_index >= EDGE_COUNT) { DTHROW("Edge index exceeds edge count, unable to get property '%s'.\n", property_name); return {}; } #endif // get the value from curve using the message index. T value = detail::curve::DeviceCurve::getEnvironmentDirectedGraphEdgeArrayProperty(_parent.graph_hash, property_name, edge_index, element_index); return value; } __device__ __forceinline__ unsigned int DeviceEnvironmentDirectedGraph::OutEdgeFilter::Edge::getEdgeDestination() const { return getProperty("_source_dest", 0); } __device__ __forceinline__ unsigned int DeviceEnvironmentDirectedGraph::InEdgeFilter::Edge::getEdgeSource() const { return getProperty("_source_dest", 1); } __device__ __forceinline__ id_t DeviceEnvironmentDirectedGraph::getVertexID(const unsigned int vertex_index) const { #if !defined(FLAMEGPU_SEATBELTS) || FLAMEGPU_SEATBELTS const unsigned int VERTEX_COUNT = detail::curve::DeviceCurve::getVariableCount("_id", graph_hash ^ detail::curve::Curve::variableHash("_environment_directed_graph_vertex")); if (vertex_index >= VERTEX_COUNT) { DTHROW("Vertex index (%u) exceeds vertex count (%u), unable to get vertex ID.\n", vertex_index, VERTEX_COUNT); return {}; } #endif return getVertexProperty("_id", vertex_index); } __device__ __forceinline__ unsigned int DeviceEnvironmentDirectedGraph::getVertexIndex(id_t vertex_id) const { const unsigned int VERTEX_RANGE = detail::curve::DeviceCurve::getVariableCount("_index_map", graph_hash ^ detail::curve::Curve::variableHash("_environment_directed_graph_vertex")) - 1; // -1 because offset is packed at the end const unsigned int VERTEX_OFFSET = detail::curve::DeviceCurve::getEnvironmentDirectedGraphVertexProperty(graph_hash, "_index_map", VERTEX_RANGE); #if !defined(FLAMEGPU_SEATBELTS) || FLAMEGPU_SEATBELTS if (vertex_id < VERTEX_OFFSET || vertex_id - VERTEX_OFFSET >= VERTEX_RANGE) { DTHROW("Vertex ID (%u) exceeds vertex range bounds (%u <= ID <= %u), unable to get vertex index.\n", vertex_id, VERTEX_OFFSET, VERTEX_OFFSET + VERTEX_RANGE - 1); return {}; } const unsigned int index = getVertexProperty("_index_map", vertex_id - VERTEX_OFFSET); if (index == 0xffffffff) { DTHROW("Vertex ID %u is not in use, unable to get vertex index.\n", vertex_id); return {}; } return index; #else return getVertexProperty("_index_map", vertex_id - VERTEX_OFFSET); #endif } template __device__ __forceinline__ T DeviceEnvironmentDirectedGraph::getVertexProperty(const char(&property_name)[M], const unsigned int vertex_index) const { #if !defined(FLAMEGPU_SEATBELTS) || FLAMEGPU_SEATBELTS const unsigned int VERTEX_COUNT = detail::curve::DeviceCurve::getVariableCount(property_name, graph_hash ^ detail::curve::Curve::variableHash("_environment_directed_graph_vertex")); if (vertex_index >= VERTEX_COUNT) { DTHROW("Vertex index (%u) exceeds vertex count (%u), unable to get property '%s'.\n", vertex_index, VERTEX_COUNT, property_name); return {}; } #endif return detail::curve::DeviceCurve::getEnvironmentDirectedGraphVertexProperty(graph_hash, property_name, vertex_index); } template __device__ __forceinline__ T DeviceEnvironmentDirectedGraph::getVertexProperty(const char(&property_name)[M], const unsigned int vertex_index, const unsigned int element_index) const { #if !defined(FLAMEGPU_SEATBELTS) || FLAMEGPU_SEATBELTS const unsigned int VERTEX_COUNT = detail::curve::DeviceCurve::getVariableCount(property_name, graph_hash ^ detail::curve::Curve::variableHash("_environment_directed_graph_vertex")); if (vertex_index >= VERTEX_COUNT) { DTHROW("Vertex index (%u) exceeds vertex count (%u), unable to get property '%s'.\n", vertex_index, VERTEX_COUNT, property_name); return {}; } #endif return detail::curve::DeviceCurve::getEnvironmentDirectedGraphVertexArrayProperty(graph_hash, property_name, vertex_index, element_index); } __device__ __forceinline__ id_t DeviceEnvironmentDirectedGraph::getEdgeSource(const unsigned int edge_index) const { #if !defined(FLAMEGPU_SEATBELTS) || FLAMEGPU_SEATBELTS const unsigned int EDGE_COUNT = detail::curve::DeviceCurve::getVariableCount("_source_dest", graph_hash ^ detail::curve::Curve::variableHash("_environment_directed_graph_edge")); if (edge_index >= EDGE_COUNT) { DTHROW("Edge index (%u) exceeds edge count (%u), unable to get edge source vertex.\n", edge_index, EDGE_COUNT); return {}; } #endif return getEdgeProperty("_source_dest", edge_index, 1); } __device__ __forceinline__ id_t DeviceEnvironmentDirectedGraph::getEdgeDestination(const unsigned int edge_index) const { #if !defined(FLAMEGPU_SEATBELTS) || FLAMEGPU_SEATBELTS const unsigned int EDGE_COUNT = detail::curve::DeviceCurve::getVariableCount("_source_dest", graph_hash ^ detail::curve::Curve::variableHash("_environment_directed_graph_edge")); if (edge_index >= EDGE_COUNT) { DTHROW("Edge index (%u) exceeds edge count (%u), unable to get edge destination vertex.\n", edge_index, EDGE_COUNT); return {}; } #endif return getEdgeProperty("_source_dest", edge_index, 0); } __device__ __forceinline__ unsigned int DeviceEnvironmentDirectedGraph::getEdgeIndex(unsigned int source_vertex_index, unsigned int destination_vertex_index) const { #if !defined(FLAMEGPU_SEATBELTS) || FLAMEGPU_SEATBELTS const unsigned int VERTEX_COUNT = detail::curve::DeviceCurve::getVariableCount("_id", graph_hash ^ detail::curve::Curve::variableHash("_environment_directed_graph_vertex")); if (source_vertex_index >= VERTEX_COUNT) { DTHROW("Source vertex index (%u) exceeds vertex count (%u), unable to get edge index.\n", source_vertex_index, VERTEX_COUNT); return {}; } else if (destination_vertex_index >= VERTEX_COUNT) { DTHROW("Destination vertex index (%u) exceeds vertex count (%u), unable to get edge index.\n", destination_vertex_index, VERTEX_COUNT); return {}; } #endif unsigned int* pbm = detail::curve::DeviceCurve::getEnvironmentDirectedGraphPBM(graph_hash); #if !defined(FLAMEGPU_SEATBELTS) || FLAMEGPU_SEATBELTS if (!pbm) { DTHROW("Graph PBM was not found.\n"); return { }; } #endif const unsigned int bucket_begin = pbm[source_vertex_index]; const unsigned int bucket_end = pbm[source_vertex_index + 1]; for (unsigned int i = bucket_begin; i < bucket_end; ++i) { // Fetch edge destinations until we find a match const unsigned int t_dest_index = detail::curve::DeviceCurve::getEnvironmentDirectedGraphEdgeArrayProperty(graph_hash, "_source_dest", i, 0); if (t_dest_index == destination_vertex_index) return i; } return { }; } template __device__ __forceinline__ T DeviceEnvironmentDirectedGraph::getEdgeProperty(const char(&property_name)[M], const unsigned int edge_index) const { #if !defined(FLAMEGPU_SEATBELTS) || FLAMEGPU_SEATBELTS const unsigned int EDGE_COUNT = detail::curve::DeviceCurve::getVariableCount("_source_dest", graph_hash ^ detail::curve::Curve::variableHash("_environment_directed_graph_edge")); if (edge_index >= EDGE_COUNT) { DTHROW("Edge index (%u) exceeds edge count (%u), unable to get property '%s'.\n", edge_index, EDGE_COUNT, property_name); return {}; } #endif return detail::curve::DeviceCurve::getEnvironmentDirectedGraphEdgeProperty(graph_hash, property_name, edge_index); } template __device__ __forceinline__ T DeviceEnvironmentDirectedGraph::getEdgeProperty(const char(&property_name)[M], const unsigned int edge_index, const unsigned int element_index) const { #if !defined(FLAMEGPU_SEATBELTS) || FLAMEGPU_SEATBELTS const unsigned int EDGE_COUNT = detail::curve::DeviceCurve::getVariableCount("_source_dest", graph_hash ^ detail::curve::Curve::variableHash("_environment_directed_graph_edge")); if (edge_index >= EDGE_COUNT) { DTHROW("Edge index (%u) exceeds edge count (%u), unable to get property '%s'.\n", edge_index, EDGE_COUNT, property_name); return {}; } #endif return detail::curve::DeviceCurve::getEnvironmentDirectedGraphEdgeArrayProperty(graph_hash, property_name, edge_index, element_index); } } // namespace flamegpu #endif // INCLUDE_FLAMEGPU_RUNTIME_ENVIRONMENT_DEVICEENVIRONMENTDIRECTEDGRAPH_CUH_