/*************************************************************************** * blitz/memblock.h MemoryBlock and MemoryBlockReference * * $Id: memblock.h,v 1.1.1.1 1999-04-09 17:58:59 ansari Exp $ * * Copyright (C) 1997,1998 Todd Veldhuizen * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU General Public License * as published by the Free Software Foundation; either version 2 * of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * Suggestions: blitz-suggest@cybervision.com * Bugs: blitz-bugs@cybervision.com * * For more information, please see the Blitz++ Home Page: * http://seurat.uwaterloo.ca/blitz/ * *************************************************************************** * $Log: not supported by cvs2svn $ * Revision 1.7 1998/06/15 16:07:01 tveldhui * When a memory block is created from an existing block of data, * add an additional reference count so that makeUnique() will * create a copy of the data. * * Revision 1.6 1998/03/14 00:04:47 tveldhui * 0.2-alpha-05 * * Revision 1.5 1997/07/16 14:51:20 tveldhui * Update: Alpha release 0.2 (Arrays) * * Revision 1.4 1997/01/24 14:42:00 tveldhui * Periodic RCS update * * Revision 1.3 1997/01/23 03:28:28 tveldhui * Periodic RCS update * * Revision 1.2 1997/01/13 22:19:58 tveldhui * Periodic RCS update * * Revision 1.1 1996/11/11 17:29:13 tveldhui * Initial revision * * *************************************************************************** * */ #ifndef __BZ_MEMBLOCK_H__ #define __BZ_MEMBLOCK_H__ #ifndef BZ_BLITZ_H #include #endif #include // ptrdiff_t BZ_NAMESPACE(blitz) // Forward declaration of MemoryBlockReference template class MemoryBlockReference; // Class MemoryBlock provides a reference-counted block of memory. This block // may be referred to by multiple vector, matrix and array objects. The memory // is automatically deallocated when the last referring object is destructed. // MemoryBlock may be subclassed to provide special allocators. template class MemoryBlock { friend class MemoryBlockReference; public: typedef P_type T_type; protected: MemoryBlock() { length_ = 0; data_ = 0; dataBlockAddress_ = 0; references_ = 0; ownData_ = 0; } _bz_explicit MemoryBlock(size_t items) { length_ = items; allocate(length_); #ifdef BZ_DEBUG_LOG_ALLOCATIONS cout << "MemoryBlock: allocated " << setw(8) << length_ << " at " << ((void *)dataBlockAddress_) << endl; #endif BZASSERT(dataBlockAddress_ != 0); references_ = 0; ownData_ = 1; } MemoryBlock(size_t length, T_type* _bz_restrict data) { length_ = length; data_ = data; dataBlockAddress_ = 0; references_ = 0; ownData_ = 0; } virtual ~MemoryBlock() { if ((dataBlockAddress_) && (ownData_)) { #ifdef BZ_DEBUG_LOG_ALLOCATIONS cout << "MemoryBlock: freed " << setw(8) << length_ << " at " << ((void *)dataBlockAddress_) << endl; #endif delete [] dataBlockAddress_; } } void addReference() { ++references_; #ifdef BZ_DEBUG_LOG_REFERENCES cout << "MemoryBlock: reffed " << setw(8) << length_ << " at " << ((void *)dataBlockAddress_) << " (r=" << references_ << ")" << endl; #endif } T_type* _bz_restrict data() { return data_; } const T_type* _bz_restrict data() const { return data_; } size_t length() const { return length_; } void removeReference() { --references_; #ifdef BZ_DEBUG_LOG_REFERENCES cout << "MemoryBlock: dereffed " << setw(8) << length_ << " at " << ((void *)dataBlockAddress_) << " (r=" << references_ << ")" << endl; #endif if (!references_) delete this; } int references() const { return references_; } protected: inline void allocate(int length); private: // Disabled member functions MemoryBlock(const MemoryBlock&) { } void operator=(const MemoryBlock&) { } private: // Data members T_type * _bz_restrict data_; T_type * _bz_restrict dataBlockAddress_; int references_; size_t length_; int ownData_; }; template class NullMemoryBlock : public MemoryBlock { public: NullMemoryBlock() { // This ensures that the delete operator will not be invoked // on an instance of NullMemoryBlock in removeReference(). this->addReference(); } virtual ~NullMemoryBlock() { } }; template class MemoryBlockReference { public: typedef P_type T_type; MemoryBlockReference() { block_ = &nullBlock_; block_->addReference(); #ifndef BZ_TEMPLATE_STATIC_INIT // This compiler doesn't handle static initialization // and templates properly. Give the NullMemoryBlock // another reference to make sure it doesn't delete // itself and cause a seg fault when we dereference it. block_->addReference(); #endif data_ = 0; } MemoryBlockReference(MemoryBlockReference& ref) { block_ = ref.block_; block_->addReference(); data_ = block_->data(); } MemoryBlockReference(MemoryBlockReference& ref, size_t offset) { block_ = ref.block_; block_->addReference(); data_ = block_->data() + offset; } MemoryBlockReference(size_t length, T_type* data) { // Create a memory block using already allocated memory. Note // that the memory will not be freed when no longer being // referenced. block_ = new MemoryBlock(length, data); block_->addReference(); // This extra reference indicates that another object somewhere // has a pointer to the data. This is necessary so that makeUnique() // will create a copy of the data. -- TV 19980615 block_->addReference(); data_ = data; } _bz_explicit MemoryBlockReference(size_t items) { block_ = new MemoryBlock(items); block_->addReference(); data_ = block_->data(); } ~MemoryBlockReference() { block_->removeReference(); } int numReferences() const { return block_->references(); } protected: void changeToNullBlock() { block_->removeReference(); block_ = &nullBlock_; block_->addReference(); data_ = 0; } void changeBlock(MemoryBlockReference& ref, size_t offset) { block_->removeReference(); block_ = ref.block_; block_->addReference(); data_ = block_->data() + offset; } void newBlock(size_t items) { block_->removeReference(); block_ = new MemoryBlock(items); block_->addReference(); data_ = block_->data(); } private: void operator=(const MemoryBlockReference&) { } private: MemoryBlock* block_; static NullMemoryBlock nullBlock_; protected: T_type * _bz_restrict data_; }; BZ_NAMESPACE_END #include #endif // __BZ_MEMBLOCK_H__