X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/5a1cad6ea487130bc074d858368cd93241ae2cf2..a949e8fac2666418b2a6eb15e94959563cf8aad6:/src/common/dynarray.cpp diff --git a/src/common/dynarray.cpp b/src/common/dynarray.cpp index e8efb32ecb..4d0ee51d28 100644 --- a/src/common/dynarray.cpp +++ b/src/common/dynarray.cpp @@ -33,6 +33,11 @@ #define max(a, b) (((a) > (b)) ? (a) : (b)) #endif +// we cast the value to long from which we cast it to void * in IndexForInsert: +// this can't work if the pointers are not big enough +wxCOMPILE_TIME_ASSERT( sizeof(long) <= sizeof(void *), + wxArraySizeOfPtrLessSizeOfLong ); // < 32 symbols + // ============================================================================ // constants // ============================================================================ @@ -48,7 +53,7 @@ // wxBaseArray - dynamic array of 'T's // ---------------------------------------------------------------------------- -#define _DECLARE_BASEARRAY(T, name) \ +#define _WX_DEFINE_BASEARRAY(T, name) \ /* ctor */ \ name::name() \ { \ @@ -102,25 +107,30 @@ name& name::operator=(const name& src) \ } \ \ /* grow the array */ \ -void name::Grow() \ +void name::Grow(size_t nIncrement) \ { \ /* only do it if no more place */ \ - if( m_nCount == m_nSize ) { \ + if( (m_nCount == m_nSize) || ((m_nSize - m_nCount) < nIncrement) ) { \ if( m_nSize == 0 ) { \ - /* was empty, alloc some memory */ \ - m_pItems = new T[WX_ARRAY_DEFAULT_INITIAL_SIZE]; \ + /* was empty, determine initial size */ \ + size_t size = WX_ARRAY_DEFAULT_INITIAL_SIZE; \ + if (size < nIncrement) size = nIncrement; \ + /* allocate some memory */ \ + m_pItems = new T[size]; \ /* only grow if allocation succeeded */ \ if ( m_pItems ) { \ - m_nSize = WX_ARRAY_DEFAULT_INITIAL_SIZE; \ + m_nSize = size; \ } \ } \ else \ { \ - /* add 50% but not too much */ \ - size_t nIncrement = m_nSize < WX_ARRAY_DEFAULT_INITIAL_SIZE \ - ? WX_ARRAY_DEFAULT_INITIAL_SIZE : m_nSize >> 1; \ - if ( nIncrement > ARRAY_MAXSIZE_INCREMENT ) \ - nIncrement = ARRAY_MAXSIZE_INCREMENT; \ + /* add at least 50% but not too much */ \ + size_t ndefIncrement = m_nSize < WX_ARRAY_DEFAULT_INITIAL_SIZE \ + ? WX_ARRAY_DEFAULT_INITIAL_SIZE : m_nSize >> 1; \ + if ( ndefIncrement > ARRAY_MAXSIZE_INCREMENT ) \ + ndefIncrement = ARRAY_MAXSIZE_INCREMENT; \ + if ( nIncrement < ndefIncrement ) \ + nIncrement = ndefIncrement; \ T *pNew = new T[m_nSize + nIncrement]; \ /* only grow if allocation succeeded */ \ if ( pNew ) { \ @@ -217,7 +227,8 @@ size_t name::IndexForInsert(T lItem, CMPFUNC fnCompare) const \ while ( lo < hi ) { \ i = (lo + hi)/2; \ \ - res = (*fnCompare)((const void *)&lItem, (const void *)&(m_pItems[i])); \ + res = (*fnCompare)((const void *)(long)lItem, \ + (const void *)(long)(m_pItems[i])); \ if ( res < 0 ) \ hi = i; \ else if ( res > 0 ) \ @@ -240,10 +251,13 @@ int name::Index(T lItem, CMPFUNC fnCompare) const \ } \ \ /* add item at the end */ \ -void name::Add(T lItem) \ +void name::Add(T lItem, size_t nInsert) \ { \ - Grow(); \ - m_pItems[m_nCount++] = lItem; \ + if (nInsert == 0) \ + return; \ + Grow(nInsert); \ + for (size_t i = 0; i < nInsert; i++) \ + m_pItems[m_nCount++] = lItem; \ } \ \ /* add item assuming the array is sorted with fnCompare function */ \ @@ -253,26 +267,33 @@ void name::Add(T lItem, CMPFUNC fnCompare) \ } \ \ /* add item at the given position */ \ -void name::Insert(T lItem, size_t nIndex) \ +void name::Insert(T lItem, size_t nIndex, size_t nInsert) \ { \ wxCHECK_RET( nIndex <= m_nCount, wxT("bad index in wxArray::Insert") ); \ + wxCHECK_RET( m_nCount <= m_nCount + nInsert, \ + wxT("array size overflow in wxArray::Insert") ); \ \ - Grow(); \ + if (nInsert == 0) \ + return; \ + Grow(nInsert); \ \ - memmove(&m_pItems[nIndex + 1], &m_pItems[nIndex], \ + memmove(&m_pItems[nIndex + nInsert], &m_pItems[nIndex], \ (m_nCount - nIndex)*sizeof(T)); \ - m_pItems[nIndex] = lItem; \ - m_nCount++; \ + for (size_t i = 0; i < nInsert; i++) \ + m_pItems[nIndex + i] = lItem; \ + m_nCount += nInsert; \ } \ \ /* removes item from array (by index) */ \ -void name::RemoveAt(size_t nIndex) \ +void name::RemoveAt(size_t nIndex, size_t nRemove) \ { \ - wxCHECK_RET( nIndex <= m_nCount, wxT("bad index in wxArray::RemoveAt") ); \ + wxCHECK_RET( nIndex < m_nCount, wxT("bad index in wxArray::RemoveAt") ); \ + wxCHECK_RET( nIndex + nRemove <= m_nCount, \ + wxT("removing too many elements in wxArray::RemoveAt") ); \ \ - memmove(&m_pItems[nIndex], &m_pItems[nIndex + 1], \ - (m_nCount - nIndex - 1)*sizeof(T)); \ - m_nCount--; \ + memmove(&m_pItems[nIndex], &m_pItems[nIndex + nRemove], \ + (m_nCount - nIndex - nRemove)*sizeof(T)); \ + m_nCount -= nRemove; \ } \ \ /* removes item from array (by value) */ \ @@ -292,9 +313,9 @@ void name::Sort(CMPFUNC fCmp) \ qsort(m_pItems, m_nCount, sizeof(T), fCmp); \ } -_DECLARE_BASEARRAY(void *, wxBaseArrayPtrVoid) -_DECLARE_BASEARRAY(short, wxBaseArrayShort) -_DECLARE_BASEARRAY(int, wxBaseArrayInt) -_DECLARE_BASEARRAY(long, wxBaseArrayLong) -_DECLARE_BASEARRAY(double, wxBaseArrayDouble) +_WX_DEFINE_BASEARRAY(const void *, wxBaseArrayPtrVoid) +_WX_DEFINE_BASEARRAY(short, wxBaseArrayShort) +_WX_DEFINE_BASEARRAY(int, wxBaseArrayInt) +_WX_DEFINE_BASEARRAY(long, wxBaseArrayLong) +//_WX_DEFINE_BASEARRAY(double, wxBaseArrayDouble)