#ifndef _LIST_H #define _LIST_H typedef struct { int count; //Number of element in the list. size_t size; //Size of list in bytes (used for type checking). void* data; //Pointer to the list contents. Must be deleted by the caller using BridgeFree (or BridgeList::Free). } ListInfo; #define ListOf(Type) ListInfo* #ifdef __cplusplus #include /** \brief A list object. This object is NOT thread safe. \tparam Type BridgeList contents type. */ template class BridgeList { public: /** \brief BridgeList constructor. \param _freeData (Optional) the free function. */ explicit BridgeList() { memset(&_listInfo, 0, sizeof(_listInfo)); } /** \brief BridgeList destructor. */ ~BridgeList() { Cleanup(); } /** \brief Gets the list data. \return Returns ListInfo->data. Can be null if the list was never initialized. Will be destroyed once this object goes out of scope! */ Type* Data() const { return reinterpret_cast(_listInfo.data); } /** \brief Gets the number of elements in the list. This will crash the program if the data is not consistent with the specified template argument. \return The number of elements in the list. */ int Count() const { if(_listInfo.size != _listInfo.count * sizeof(Type)) //make sure the user is using the correct type. __debugbreak(); return _listInfo.count; } /** \brief Cleans up the list, freeing the list data when it is not null. */ void Cleanup() { if(_listInfo.data) { BridgeFree(_listInfo.data); _listInfo.data = nullptr; } } /** \brief Reference operator (cleans up the previous list) \return Pointer to the ListInfo. */ ListInfo* operator&() { Cleanup(); return &_listInfo; } /** \brief Array indexer operator. This will crash if you try to access out-of-bounds. \param index Zero-based index of the item you want to get. \return Reference to a value at that index. */ Type & operator[](size_t index) const { if(index >= size_t(Count())) //make sure the out-of-bounds access is caught as soon as possible. __debugbreak(); return Data()[index]; } /** \brief Copies data to a ListInfo structure.. \param [out] listInfo If non-null, information describing the list. \param listData Data to copy in the ListInfo structure. \return true if it succeeds, false if it fails. */ static bool CopyData(ListInfo* listInfo, const std::vector & listData) { if(!listInfo) return false; listInfo->count = int(listData.size()); listInfo->size = listInfo->count * sizeof(Type); if(listInfo->count) { listInfo->data = BridgeAlloc(listInfo->size); Type* curItem = reinterpret_cast(listInfo->data); for(const auto & item : listData) { *curItem = item; ++curItem; } } else listInfo->data = nullptr; return true; } static bool Free(const ListInfo* listInfo) { if(!listInfo || listInfo->size != listInfo->count * sizeof(Type) || (listInfo->count && !listInfo->data)) return false; BridgeFree(listInfo->data); return true; } static bool ToVector(const ListInfo* listInfo, std::vector & listData, bool freedata = true) { if(!listInfo || listInfo->size != listInfo->count * sizeof(Type) || (listInfo->count && !listInfo->data)) return false; listData.resize(listInfo->count); for(int i = 0; i < listInfo->count; i++) listData[i] = ((Type*)listInfo->data)[i]; if(freedata && listInfo->data) BridgeFree(listInfo->data); return true; } private: ListInfo _listInfo; }; #endif //__cplusplus #endif //_LIST_H