Team Fortress 2 Source Code as on 22/4/2020
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

350 lines
9.7 KiB

  1. //========= Copyright Valve Corporation, All rights reserved. ============//
  2. //
  3. // Purpose:
  4. //
  5. // $NoKeywords: $
  6. //
  7. // A growable memory class.
  8. //===========================================================================//
  9. #ifndef UTLBLOCKMEMORY_H
  10. #define UTLBLOCKMEMORY_H
  11. #ifdef _WIN32
  12. #pragma once
  13. #endif
  14. #include "tier0/dbg.h"
  15. #include "tier0/platform.h"
  16. #include "mathlib/mathlib.h"
  17. #include "tier0/memalloc.h"
  18. #include "tier0/memdbgon.h"
  19. #pragma warning (disable:4100)
  20. #pragma warning (disable:4514)
  21. //-----------------------------------------------------------------------------
  22. #ifdef UTBLOCKLMEMORY_TRACK
  23. #define UTLBLOCKMEMORY_TRACK_ALLOC() MemAlloc_RegisterAllocation( "Sum of all UtlBlockMemory", 0, NumAllocated() * sizeof(T), NumAllocated() * sizeof(T), 0 )
  24. #define UTLBLOCKMEMORY_TRACK_FREE() if ( !m_pMemory ) ; else MemAlloc_RegisterDeallocation( "Sum of all UtlBlockMemory", 0, NumAllocated() * sizeof(T), NumAllocated() * sizeof(T), 0 )
  25. #else
  26. #define UTLBLOCKMEMORY_TRACK_ALLOC() ((void)0)
  27. #define UTLBLOCKMEMORY_TRACK_FREE() ((void)0)
  28. #endif
  29. //-----------------------------------------------------------------------------
  30. // The CUtlBlockMemory class:
  31. // A growable memory class that allocates non-sequential blocks, but is indexed sequentially
  32. //-----------------------------------------------------------------------------
  33. template< class T, class I >
  34. class CUtlBlockMemory
  35. {
  36. public:
  37. // constructor, destructor
  38. CUtlBlockMemory( int nGrowSize = 0, int nInitSize = 0 );
  39. ~CUtlBlockMemory();
  40. // Set the size by which the memory grows - round up to the next power of 2
  41. void Init( int nGrowSize = 0, int nInitSize = 0 );
  42. // here to match CUtlMemory, but only used by ResetDbgInfo, so it can just return NULL
  43. T* Base() { return NULL; }
  44. const T* Base() const { return NULL; }
  45. class Iterator_t
  46. {
  47. public:
  48. Iterator_t( I i ) : index( i ) {}
  49. I index;
  50. bool operator==( const Iterator_t it ) const { return index == it.index; }
  51. bool operator!=( const Iterator_t it ) const { return index != it.index; }
  52. };
  53. Iterator_t First() const { return Iterator_t( IsIdxValid( 0 ) ? 0 : InvalidIndex() ); }
  54. Iterator_t Next( const Iterator_t &it ) const { return Iterator_t( IsIdxValid( it.index + 1 ) ? it.index + 1 : InvalidIndex() ); }
  55. I GetIndex( const Iterator_t &it ) const { return it.index; }
  56. bool IsIdxAfter( I i, const Iterator_t &it ) const { return i > it.index; }
  57. bool IsValidIterator( const Iterator_t &it ) const { return IsIdxValid( it.index ); }
  58. Iterator_t InvalidIterator() const { return Iterator_t( InvalidIndex() ); }
  59. // element access
  60. T& operator[]( I i );
  61. const T& operator[]( I i ) const;
  62. T& Element( I i );
  63. const T& Element( I i ) const;
  64. // Can we use this index?
  65. bool IsIdxValid( I i ) const;
  66. static I InvalidIndex() { return ( I )-1; }
  67. void Swap( CUtlBlockMemory< T, I > &mem );
  68. // Size
  69. int NumAllocated() const;
  70. int Count() const { return NumAllocated(); }
  71. // Grows memory by max(num,growsize) rounded up to the next power of 2, and returns the allocation index/ptr
  72. void Grow( int num = 1 );
  73. // Makes sure we've got at least this much memory
  74. void EnsureCapacity( int num );
  75. // Memory deallocation
  76. void Purge();
  77. // Purge all but the given number of elements
  78. void Purge( int numElements );
  79. protected:
  80. int Index( int major, int minor ) const { return ( major << m_nIndexShift ) | minor; }
  81. int MajorIndex( int i ) const { return i >> m_nIndexShift; }
  82. int MinorIndex( int i ) const { return i & m_nIndexMask; }
  83. void ChangeSize( int nBlocks );
  84. int NumElementsInBlock() const { return m_nIndexMask + 1; }
  85. T** m_pMemory;
  86. int m_nBlocks;
  87. int m_nIndexMask : 27;
  88. int m_nIndexShift : 5;
  89. };
  90. //-----------------------------------------------------------------------------
  91. // constructor, destructor
  92. //-----------------------------------------------------------------------------
  93. template< class T, class I >
  94. CUtlBlockMemory<T,I>::CUtlBlockMemory( int nGrowSize, int nInitAllocationCount )
  95. : m_pMemory( 0 ), m_nBlocks( 0 ), m_nIndexMask( 0 ), m_nIndexShift( 0 )
  96. {
  97. Init( nGrowSize, nInitAllocationCount );
  98. }
  99. template< class T, class I >
  100. CUtlBlockMemory<T,I>::~CUtlBlockMemory()
  101. {
  102. Purge();
  103. }
  104. //-----------------------------------------------------------------------------
  105. // Fast swap
  106. //-----------------------------------------------------------------------------
  107. template< class T, class I >
  108. void CUtlBlockMemory<T,I>::Swap( CUtlBlockMemory< T, I > &mem )
  109. {
  110. swap( m_pMemory, mem.m_pMemory );
  111. swap( m_nBlocks, mem.m_nBlocks );
  112. swap( m_nIndexMask, mem.m_nIndexMask );
  113. swap( m_nIndexShift, mem.m_nIndexShift );
  114. }
  115. //-----------------------------------------------------------------------------
  116. // Set the size by which the memory grows - round up to the next power of 2
  117. //-----------------------------------------------------------------------------
  118. template< class T, class I >
  119. void CUtlBlockMemory<T,I>::Init( int nGrowSize /* = 0 */, int nInitSize /* = 0 */ )
  120. {
  121. Purge();
  122. if ( nGrowSize == 0)
  123. {
  124. // default grow size is smallest size s.t. c++ allocation overhead is ~6% of block size
  125. nGrowSize = ( 127 + sizeof( T ) ) / sizeof( T );
  126. }
  127. nGrowSize = SmallestPowerOfTwoGreaterOrEqual( nGrowSize );
  128. m_nIndexMask = nGrowSize - 1;
  129. m_nIndexShift = 0;
  130. while ( nGrowSize > 1 )
  131. {
  132. nGrowSize >>= 1;
  133. ++m_nIndexShift;
  134. }
  135. Assert( m_nIndexMask + 1 == ( 1 << m_nIndexShift ) );
  136. Grow( nInitSize );
  137. }
  138. //-----------------------------------------------------------------------------
  139. // element access
  140. //-----------------------------------------------------------------------------
  141. template< class T, class I >
  142. inline T& CUtlBlockMemory<T,I>::operator[]( I i )
  143. {
  144. Assert( IsIdxValid(i) );
  145. T *pBlock = m_pMemory[ MajorIndex( i ) ];
  146. return pBlock[ MinorIndex( i ) ];
  147. }
  148. template< class T, class I >
  149. inline const T& CUtlBlockMemory<T,I>::operator[]( I i ) const
  150. {
  151. Assert( IsIdxValid(i) );
  152. const T *pBlock = m_pMemory[ MajorIndex( i ) ];
  153. return pBlock[ MinorIndex( i ) ];
  154. }
  155. template< class T, class I >
  156. inline T& CUtlBlockMemory<T,I>::Element( I i )
  157. {
  158. Assert( IsIdxValid(i) );
  159. T *pBlock = m_pMemory[ MajorIndex( i ) ];
  160. return pBlock[ MinorIndex( i ) ];
  161. }
  162. template< class T, class I >
  163. inline const T& CUtlBlockMemory<T,I>::Element( I i ) const
  164. {
  165. Assert( IsIdxValid(i) );
  166. const T *pBlock = m_pMemory[ MajorIndex( i ) ];
  167. return pBlock[ MinorIndex( i ) ];
  168. }
  169. //-----------------------------------------------------------------------------
  170. // Size
  171. //-----------------------------------------------------------------------------
  172. template< class T, class I >
  173. inline int CUtlBlockMemory<T,I>::NumAllocated() const
  174. {
  175. return m_nBlocks * NumElementsInBlock();
  176. }
  177. //-----------------------------------------------------------------------------
  178. // Is element index valid?
  179. //-----------------------------------------------------------------------------
  180. template< class T, class I >
  181. inline bool CUtlBlockMemory<T,I>::IsIdxValid( I i ) const
  182. {
  183. return ( i >= 0 ) && ( MajorIndex( i ) < m_nBlocks );
  184. }
  185. template< class T, class I >
  186. void CUtlBlockMemory<T,I>::Grow( int num )
  187. {
  188. if ( num <= 0 )
  189. return;
  190. int nBlockSize = NumElementsInBlock();
  191. int nBlocks = ( num + nBlockSize - 1 ) / nBlockSize;
  192. ChangeSize( m_nBlocks + nBlocks );
  193. }
  194. template< class T, class I >
  195. void CUtlBlockMemory<T,I>::ChangeSize( int nBlocks )
  196. {
  197. UTLBLOCKMEMORY_TRACK_FREE(); // this must stay before the recalculation of m_nBlocks, since it implicitly uses the old value
  198. int nBlocksOld = m_nBlocks;
  199. m_nBlocks = nBlocks;
  200. UTLBLOCKMEMORY_TRACK_ALLOC(); // this must stay after the recalculation of m_nBlocks, since it implicitly uses the new value
  201. if ( m_pMemory )
  202. {
  203. // free old blocks if shrinking
  204. // Only possible if m_pMemory is non-NULL (and avoids PVS-Studio warning)
  205. for ( int i = m_nBlocks; i < nBlocksOld; ++i )
  206. {
  207. UTLBLOCKMEMORY_TRACK_FREE();
  208. free( (void*)m_pMemory[ i ] );
  209. }
  210. MEM_ALLOC_CREDIT_CLASS();
  211. m_pMemory = (T**)realloc( m_pMemory, m_nBlocks * sizeof(T*) );
  212. Assert( m_pMemory );
  213. }
  214. else
  215. {
  216. MEM_ALLOC_CREDIT_CLASS();
  217. m_pMemory = (T**)malloc( m_nBlocks * sizeof(T*) );
  218. Assert( m_pMemory );
  219. }
  220. if ( !m_pMemory )
  221. {
  222. Error( "CUtlBlockMemory overflow!\n" );
  223. }
  224. // allocate new blocks if growing
  225. int nBlockSize = NumElementsInBlock();
  226. for ( int i = nBlocksOld; i < m_nBlocks; ++i )
  227. {
  228. MEM_ALLOC_CREDIT_CLASS();
  229. m_pMemory[ i ] = (T*)malloc( nBlockSize * sizeof( T ) );
  230. Assert( m_pMemory[ i ] );
  231. }
  232. }
  233. //-----------------------------------------------------------------------------
  234. // Makes sure we've got at least this much memory
  235. //-----------------------------------------------------------------------------
  236. template< class T, class I >
  237. inline void CUtlBlockMemory<T,I>::EnsureCapacity( int num )
  238. {
  239. Grow( num - NumAllocated() );
  240. }
  241. //-----------------------------------------------------------------------------
  242. // Memory deallocation
  243. //-----------------------------------------------------------------------------
  244. template< class T, class I >
  245. void CUtlBlockMemory<T,I>::Purge()
  246. {
  247. if ( !m_pMemory )
  248. return;
  249. for ( int i = 0; i < m_nBlocks; ++i )
  250. {
  251. UTLBLOCKMEMORY_TRACK_FREE();
  252. free( (void*)m_pMemory[ i ] );
  253. }
  254. m_nBlocks = 0;
  255. UTLBLOCKMEMORY_TRACK_FREE();
  256. free( (void*)m_pMemory );
  257. m_pMemory = 0;
  258. }
  259. template< class T, class I >
  260. void CUtlBlockMemory<T,I>::Purge( int numElements )
  261. {
  262. Assert( numElements >= 0 );
  263. int nAllocated = NumAllocated();
  264. if ( numElements > nAllocated )
  265. {
  266. // Ensure this isn't a grow request in disguise.
  267. Assert( numElements <= nAllocated );
  268. return;
  269. }
  270. if ( numElements <= 0 )
  271. {
  272. Purge();
  273. return;
  274. }
  275. int nBlockSize = NumElementsInBlock();
  276. int nBlocksOld = m_nBlocks;
  277. int nBlocks = ( numElements + nBlockSize - 1 ) / nBlockSize;
  278. // If the number of blocks is the same as the allocated number of blocks, we are done.
  279. if ( nBlocks == m_nBlocks )
  280. return;
  281. ChangeSize( nBlocks );
  282. }
  283. #include "tier0/memdbgoff.h"
  284. #endif // UTLBLOCKMEMORY_H