26 #ifndef _BLOCKSTORE_H_ 27 #define _BLOCKSTORE_H_ 33 #include <amps/ampscrc.hpp> 34 #if __cplusplus >= 201103L || _MSC_VER >= 1900 40 #include <sys/timeb.h> 65 DEFAULT_BLOCK_HEADER_SIZE = 32,
66 DEFAULT_BLOCKS_PER_REALLOC = 1000,
67 DEFAULT_BLOCK_SIZE = 2048
78 amps_uint64_t _sequence;
85 Block(
size_t offset_) : _offset(offset_), _sequence(0)
86 , _nextInChain(0), _nextInList(0)
91 Block() : _offset(0), _sequence(0)
92 , _nextInChain(0), _nextInList((
Block*)(
this + 1))
96 Block* init(
size_t index_, amps_uint32_t blockSize_)
98 _offset = index_ * blockSize_;
103 Block* setOffset(
size_t offset_)
113 typedef Lock<Mutex> BufferLock;
114 typedef Unlock<Mutex> BufferUnlock;
115 typedef bool (*ResizeHandler)(size_t,
void*);
116 typedef std::vector<Block*> BlockList;
135 amps_uint32_t blocksPerRealloc_ = DEFAULT_BLOCKS_PER_REALLOC,
136 amps_uint32_t blockHeaderSize_ = DEFAULT_BLOCK_HEADER_SIZE,
137 amps_uint32_t blockSize_ = DEFAULT_BLOCK_SIZE)
138 : _buffer(buffer_), _freeList(0), _usedList(0)
139 , _endOfUsedList(0), _blocksPerRealloc(blocksPerRealloc_)
140 , _blockSize(blockSize_), _blockHeaderSize(blockHeaderSize_)
141 , _blocksAvailable(0), _resizeHandler(0), _resizeUserData(0)
150 for (BlockList::iterator i = _blockList.begin();
151 i != _blockList.end(); ++i)
169 return _blockHeaderSize;
208 return _lock.wait(timeout_);
222 _resizeHandler = resizeHandler_;
223 _resizeUserData = userData_;
242 return _endOfUsedList;
252 _blocksAvailable = freeCount_;
268 _endOfUsedList = block_;
277 _blockList.push_back(blockArray_);
285 Block*
get(amps_uint32_t numBlocksInChain_)
290 while (_blocksAvailable < numBlocksInChain_)
293 unsigned int blocksNeeded = numBlocksInChain_ - _blocksAvailable;
294 amps_uint32_t addedBlocks = (blocksNeeded / _blocksPerRealloc + 1)
296 size_t size = _buffer->getSize() + (addedBlocks * _blockSize);
303 for (
unsigned int i = 0; i < numBlocksInChain_; ++i)
307 _freeList = _freeList->_nextInList;
308 next->_nextInList = 0;
318 last->_nextInChain = next;
330 _endOfUsedList->_nextInList = first;
332 _endOfUsedList = first;
333 _blocksAvailable -= numBlocksInChain_;
344 assert(_endOfUsedList);
346 if (_usedList == block_)
349 _usedList = _usedList->_nextInList;
358 Block* used = _usedList;
361 if (used->_nextInList == block_)
363 used->_nextInList = block_->_nextInList;
366 used = used->_nextInList;
374 _flattenToFreeList(block_);
381 AMPS_ATOMIC_BASE_TYPE
put(amps_uint64_t sequence_)
384 assert(_endOfUsedList);
385 Block* used = _usedList;
386 AMPS_ATOMIC_BASE_TYPE removalCount = 0;
387 while (used && used->_sequence <= sequence_)
389 Block* next = used->_nextInList;
391 _flattenToFreeList(used);
410 Block* newEndOfUsedList = 0;
411 for (
Block* used = _usedList; used; used = used->_nextInList)
415 if (newEndOfUsedList)
417 newEndOfUsedList->_nextInList = 0;
423 _endOfUsedList = newEndOfUsedList;
425 newEndOfUsedList = used;
429 for (
Block* block = block_; block; block = next)
431 next = block->_nextInList;
432 _flattenToFreeList(block);
441 size_t startSize = _buffer->getSize();
445 startSize = _buffer->getSize();
448 amps_uint32_t numBlocks = (amps_uint32_t)(startSize) /
getBlockSize();
449 _freeList =
new Block[numBlocks];
450 _blockList.push_back(_freeList);
451 for (
size_t i = 0; i < numBlocks; ++i)
455 _freeList[numBlocks - 1]._nextInList = 0;
456 _blocksAvailable += numBlocks;
458 assert(_blocksAvailable);
465 return _blocksPerRealloc * _blockSize;
472 return _blocksPerRealloc;
488 if (_buffer->getSize() >= size_)
492 if (!_lock.wait(1000))
494 amps_invoke_waiting_function();
497 FlagFlip flip(&_resizing);
498 bool okToResize =
false;
501 BufferUnlock u(_lock);
503 okToResize = _canResize(size_);
512 size_t oldSize = _buffer->getSize();
513 amps_uint32_t oldBlocks = (amps_uint32_t)(oldSize /
getBlockSize());
514 if (oldSize >= size_)
519 _buffer->setSize(size_);
520 _buffer->zero(oldSize, size_ - oldSize);
522 *pNewBlocks_ = (amps_uint32_t)((size_ - oldSize) /
getBlockSize());
523 freeList =
new Block[*pNewBlocks_];
524 for (
size_t i = 0; i < *pNewBlocks_; ++i)
528 freeList[*pNewBlocks_ - 1]._nextInList = 0;
531 catch (
const std::bad_alloc&)
533 catch (
const std::bad_alloc& e)
536 std::ostringstream os;
537 os <<
"BlockStore failed to allocate " << size_
538 <<
" bytes for resize of store from " << _buffer->getSize()
540 throw StoreException(os.str());
551 amps_uint32_t newBlocks = 0;
553 if (!addedBlockList || !newBlocks)
558 _blockList.push_back(addedBlockList);
559 addedBlockList[newBlocks - 1]._nextInList = _freeList;
560 _freeList = addedBlockList;
561 _blocksAvailable += newBlocks;
562 assert(_blocksAvailable);
570 if (_usedList || _freeList)
574 amps_uint32_t oldSize = _blockSize;
575 _blockSize = blockSize_;
585 if (_usedList || _freeList)
589 amps_uint32_t oldSize = _blockHeaderSize;
590 _blockHeaderSize = blockHeaderSize_;
604 bool _canResize(
size_t requestedSize_)
608 return _resizeHandler(requestedSize_, _resizeUserData);
617 void _flattenToFreeList(
Block* block_)
620 Block* current = block_;
623 Block* chain = current->_nextInChain;
625 _buffer->zero(current->_offset, _blockHeaderSize);
627 current->_nextInList = _freeList;
630 current->_sequence = (amps_uint64_t)0;
631 current->_nextInChain = 0;
635 assert(_blocksAvailable);
645 Block* _endOfUsedList;
647 amps_uint32_t _blocksPerRealloc;
649 amps_uint32_t _blockSize;
650 amps_uint32_t _blockHeaderSize;
652 amps_uint32_t _blocksAvailable;
654 ResizeHandler _resizeHandler;
656 void* _resizeUserData;
658 BlockList _blockList;
660 #if __cplusplus >= 201103L || _MSC_VER >= 1900 661 std::atomic<bool> _resizing;
663 volatile bool _resizing;
void wait()
Wait for a signal.
Definition: BlockStore.hpp:197
void setResizeHandler(ResizeHandler resizeHandler_, void *userData_)
Set a resize handler that is called with the new total size of the Buffer.
Definition: BlockStore.hpp:220
amps_uint32_t setBlockSize(amps_uint32_t blockSize_)
Set the size to use for all Blocks.
Definition: BlockStore.hpp:568
void acquireRead() const
Acquire the lock for this object.
Definition: BlockStore.hpp:176
Constants
Default constant values for BlockStore.
Definition: BlockStore.hpp:63
Buffer * getBuffer()
Return the buffer underlying the store for direct write/read.
Definition: BlockStore.hpp:597
void put(Block *block_)
Return the given chain of Blocks to the free list for reuse.
Definition: BlockStore.hpp:341
~BlockStore()
Destructor that cleans up the buffer and other associated memory.
Definition: BlockStore.hpp:148
void releaseRead() const
Release the lock for this object. Used by RAII templates.
Definition: BlockStore.hpp:183
Block * front() const
Get the first used block in the store.
Definition: BlockStore.hpp:230
amps_uint32_t getDefaultResizeBlocks() const
Return the default number of blocks for each resize.
Definition: BlockStore.hpp:470
void addBlocks(Block *blockArray_)
Allow users to create Block arrays during recovery that are tracked for cleanup here with all other B...
Definition: BlockStore.hpp:275
bool wait(long timeout_)
Wait timeout_ ms for a signal.
Definition: BlockStore.hpp:206
Block * back() const
Get the last used block in the store.
Definition: BlockStore.hpp:240
void resize(size_t size_)
Resize the buffer to the requested size, adding all new space as unused Blocks for the free list...
Definition: BlockStore.hpp:549
Used as a base class for other stores in the AMPS C++ client, this is an implementation that breaks a...
Definition: BlockStore.hpp:58
void init()
Initialize, assuming that _buffer has no existing information.
Definition: BlockStore.hpp:439
void setEndOfUsedList(Block *block_)
Allow containing classes to initialize the used list in recovery.
Definition: BlockStore.hpp:266
size_t getDefaultResizeSize() const
Return the default number of bytes for each resize.
Definition: BlockStore.hpp:463
void setUsedList(Block *block_)
Allow containing classes to initialize the used list in recovery.
Definition: BlockStore.hpp:258
Core type, function, and class declarations for the AMPS C++ client.
Provides AMPS::Buffer, an abstract base class used by the store implementations in the AMPS client...
amps_uint32_t setBlockHeaderSize(amps_uint32_t blockHeaderSize_)
Set the size to use for the header for all Blocks.
Definition: BlockStore.hpp:583
void setFreeList(Block *block_, amps_uint32_t freeCount_)
Allow containing classes to initialize the free list in recovery.
Definition: BlockStore.hpp:249
amps_uint32_t getBlockHeaderSize() const
Get the size of a header within each Block, as set in the constructor.
Definition: BlockStore.hpp:167
Abstract base class for implementing a buffer to be used by a StoreImpl for storage of publish messag...
Definition: Buffer.hpp:40
BlockStore(Buffer *buffer_, amps_uint32_t blocksPerRealloc_=DEFAULT_BLOCKS_PER_REALLOC, amps_uint32_t blockHeaderSize_=DEFAULT_BLOCK_HEADER_SIZE, amps_uint32_t blockSize_=DEFAULT_BLOCK_SIZE)
Create a BlockStore using buffer_ and default block size, that grows by blocksPerRealloc_ blocks when...
Definition: BlockStore.hpp:134
void putAll(Block *block_)
Return all Blocks starting with the given Block to the free list.
Definition: BlockStore.hpp:407
Used as metadata for each block in a Buffer.
Definition: BlockStore.hpp:72
Block * resizeBuffer(size_t size_, amps_uint32_t *pNewBlocks_)
Resize the buffer to the requested size, returning all new space.
Definition: BlockStore.hpp:483
void signalAll()
Signal lock waiters.
Definition: BlockStore.hpp:190
amps_uint32_t getBlockSize() const
Get the size of each Block, as set in the constructor.
Definition: BlockStore.hpp:160
AMPS_ATOMIC_BASE_TYPE put(amps_uint64_t sequence_)
Return all Blocks with sequence <= sequence_ for reuse.
Definition: BlockStore.hpp:381
Definition: ampsplusplus.hpp:103