//%2005//////////////////////////////////////////////////////////////////////// // // Copyright (c) 2000, 2001, 2002 BMC Software; Hewlett-Packard Development // Company, L.P.; IBM Corp.; The Open Group; Tivoli Systems. // Copyright (c) 2003 BMC Software; Hewlett-Packard Development Company, L.P.; // IBM Corp.; EMC Corporation, The Open Group. // Copyright (c) 2004 BMC Software; Hewlett-Packard Development Company, L.P.; // IBM Corp.; EMC Corporation; VERITAS Software Corporation; The Open Group. // Copyright (c) 2005 Hewlett-Packard Development Company, L.P.; IBM Corp.; // EMC Corporation; VERITAS Software Corporation; The Open Group. // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to // deal in the Software without restriction, including without limitation the // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or // sell copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // THE ABOVE COPYRIGHT NOTICE AND THIS PERMISSION NOTICE SHALL BE INCLUDED IN // ALL COPIES OR SUBSTANTIAL PORTIONS OF THE SOFTWARE. THE SOFTWARE IS PROVIDED // "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT // LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR // PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT // HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN // ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. // //============================================================================== // // Author: Mike Brasher (mbrasher@bmc.com) // // Modified By: Roger Kumpf, Hewlett-Packard Company (roger_kumpf@hp.com) // Mike Brasher, Inova Europe (mike-brasher@austin.rr.com) // //%///////////////////////////////////////////////////////////////////////////// // Only include if not included as general template or if explicit instantiation #if !defined(Pegasus_ArrayImpl_h) || defined(PEGASUS_ARRAY_T) #if !defined(PEGASUS_ARRAY_T) #define Pegasus_ArrayImpl_h #endif PEGASUS_NAMESPACE_END #include #include #include #include PEGASUS_NAMESPACE_BEGIN PEGASUS_COMMON_LINKAGE void ArrayThrowIndexOutOfBoundsException(); #ifndef PEGASUS_ARRAY_T template #endif Array::Array() { _rep = &ArrayRepBase::_empty_rep; } #ifndef PEGASUS_ARRAY_T template #endif Array::Array(const Array& x) { _rep = x._rep; ArrayRep::ref(Array_rep); } #ifndef PEGASUS_ARRAY_T template #endif Array::Array(Uint32 size) { _rep = ArrayRep::alloc(size); if (!_rep) { throw NullPointer(); } InitializeRaw(Array_data, size); } #ifndef PEGASUS_ARRAY_T template #endif Array::Array(Uint32 size, const PEGASUS_ARRAY_T& x) { _rep = ArrayRep::alloc(size); if (!_rep) { throw NullPointer(); } PEGASUS_ARRAY_T* data = Array_data; // Note: we could use template specialization (by adding functions to // Memory.h) so that this loop becomes a memset() for single byte raw // types, but this function is rarely called. while (size--) new(data++) PEGASUS_ARRAY_T(x); } #ifndef PEGASUS_ARRAY_T template #endif Array::Array(const PEGASUS_ARRAY_T* items, Uint32 size) { _rep = ArrayRep::alloc(size); if (!_rep) { throw NullPointer(); } CopyToRaw(Array_data, items, size); } #ifndef PEGASUS_ARRAY_T template #endif Array::~Array() { ArrayRep::unref(Array_rep); } #ifndef PEGASUS_ARRAY_T template #endif Array& Array::operator=( const Array& x) { if (x._rep != Array_rep) { ArrayRep::unref(Array_rep); _rep = x._rep; ArrayRep::ref(Array_rep); } return *this; } #ifndef PEGASUS_ARRAY_T template #endif void Array::clear() { if (Array_size) { if (Array_refs.get() == 1) { Destroy(Array_data, Array_size); Array_size = 0; } else { ArrayRep::unref(Array_rep); _rep = &ArrayRepBase::_empty_rep; } } } #ifndef PEGASUS_ARRAY_T template #endif void Array::reserveCapacity(Uint32 capacity) { if (capacity > Array_capacity || Array_refs.get() != 1) { ArrayRep* rep = ArrayRep::alloc(capacity); if (!rep) return; rep->size = Array_size; if (Array_refs.get() == 1) { memcpy(rep->data(), Array_data, Array_size*sizeof(PEGASUS_ARRAY_T)); Array_size = 0; } else CopyToRaw(rep->data(), Array_data, Array_size); ArrayRep::unref(Array_rep); _rep = rep; } } #ifndef PEGASUS_ARRAY_T template #endif void Array::grow(Uint32 size, const PEGASUS_ARRAY_T& x) { reserveCapacity(Array_size + size); PEGASUS_ARRAY_T* p = Array_data + Array_size; Uint32 n = size; while (n--) new(p++) PEGASUS_ARRAY_T(x); Array_size += size; } #ifndef PEGASUS_ARRAY_T template #endif void Array::swap(Array& x) { ArrayRep* tmp = Array_rep; _rep = x._rep; x._rep = tmp; } #ifndef PEGASUS_ARRAY_T template #endif void Array::append(const PEGASUS_ARRAY_T& x) { Uint32 n = Array_size + 1; if (n > Array_capacity || Array_refs.get() != 1) reserveCapacity(n); new (Array_data + Array_size) PEGASUS_ARRAY_T(x); Array_size++; } #ifndef PEGASUS_ARRAY_T template #endif void Array::append(const PEGASUS_ARRAY_T* x, Uint32 size) { Uint32 n = Array_size + size; reserveCapacity(n); CopyToRaw(Array_data + Array_size, x, size); Array_size = n; } #ifndef PEGASUS_ARRAY_T template #endif void Array::appendArray(const Array& x) { append(x.getData(), x.size()); } #ifndef PEGASUS_ARRAY_T template #endif void Array::prepend(const PEGASUS_ARRAY_T& x) { prepend(&x, 1); } #ifndef PEGASUS_ARRAY_T template #endif void Array::prepend(const PEGASUS_ARRAY_T* x, Uint32 size) { reserveCapacity(Array_size + size); memmove( Array_data + size, Array_data, sizeof(PEGASUS_ARRAY_T) * Array_size); CopyToRaw(Array_data, x, size); Array_size += size; } #ifndef PEGASUS_ARRAY_T template #endif void Array::insert(Uint32 index, const PEGASUS_ARRAY_T& x) { insert(index, &x, 1); } #ifndef PEGASUS_ARRAY_T template #endif void Array::insert( Uint32 index, const PEGASUS_ARRAY_T* x, Uint32 size) { if (index > Array_size) { throw IndexOutOfBoundsException(); } reserveCapacity(Array_size + size); Uint32 n = Array_size - index; if (n) { memmove( Array_data + index + size, Array_data + index, sizeof(PEGASUS_ARRAY_T) * n); } CopyToRaw(Array_data + index, x, size); Array_size += size; } #ifndef PEGASUS_ARRAY_T template #endif void Array::remove(Uint32 index) { remove(index, 1); } #ifndef PEGASUS_ARRAY_T template #endif void Array::remove(Uint32 index, Uint32 size) { if (Array_refs.get() != 1) _rep = ArrayRep::copy_on_write(Array_rep); // Case 1: attempting to remove last element (this is an optimization // for when the array is used as a stack; see Stack class). if (index + 1 == Array_size) { Destroy(Array_data + index, 1); Array_size--; return; } // Case 2: not attempting to remove last element: if (index + size - 1 > Array_size) { throw IndexOutOfBoundsException(); } Destroy(Array_data + index, size); Uint32 rem = Array_size - (index + size); if (rem) { memmove( Array_data + index, Array_data + index + size, sizeof(PEGASUS_ARRAY_T) * rem); } Array_size -= size; } #ifndef PEGASUS_ARRAY_T template #endif Uint32 Array::size() const { return Array_size; } #ifndef PEGASUS_ARRAY_T template #endif PEGASUS_ARRAY_T& Array::operator[]( Uint32 index) { #ifndef PEGASUS_ARRAY_NO_THROW if (index >= Array_size) ArrayThrowIndexOutOfBoundsException(); #endif if (Array_refs.get() != 1) _rep = ArrayRep::copy_on_write(Array_rep); return Array_data[index]; } #ifndef PEGASUS_ARRAY_T template #endif const PEGASUS_ARRAY_T& Array::operator[]( Uint32 index) const { #ifndef PEGASUS_ARRAY_NO_THROW if (index >= Array_size) ArrayThrowIndexOutOfBoundsException(); #endif return Array_data[index]; } #ifndef PEGASUS_ARRAY_T template #endif Uint32 Array::getCapacity() const { return Array_capacity; } #ifndef PEGASUS_ARRAY_T template #endif const PEGASUS_ARRAY_T* Array::getData() const { return Array_data; } #ifndef PEGASUS_ARRAY_T template #endif PEGASUS_ARRAY_T* Array::_data() const { return Array_data; } #endif //!defined(Pegasus_ArrayImpl_h) || !defined(PEGASUS_ARRAY_T)