2011-07-28 14:26:00 +00:00
|
|
|
|
2008-12-17 15:59:43 +00:00
|
|
|
/*
|
2011-07-28 14:26:00 +00:00
|
|
|
* Copyright 2006 The Android Open Source Project
|
2008-12-17 15:59:43 +00:00
|
|
|
*
|
2011-07-28 14:26:00 +00:00
|
|
|
* Use of this source code is governed by a BSD-style license that can be
|
|
|
|
* found in the LICENSE file.
|
2008-12-17 15:59:43 +00:00
|
|
|
*/
|
|
|
|
|
2011-07-28 14:26:00 +00:00
|
|
|
|
2008-12-17 15:59:43 +00:00
|
|
|
#ifndef SkDeque_DEFINED
|
|
|
|
#define SkDeque_DEFINED
|
|
|
|
|
2018-07-16 21:44:06 +00:00
|
|
|
#include "../private/SkNoncopyable.h"
|
2008-12-17 15:59:43 +00:00
|
|
|
#include "SkTypes.h"
|
|
|
|
|
2012-08-17 13:53:05 +00:00
|
|
|
/*
|
|
|
|
* The deque class works by blindly creating memory space of a specified element
|
2012-08-23 18:09:54 +00:00
|
|
|
* size. It manages the memory as a doubly linked list of blocks each of which
|
|
|
|
* can contain multiple elements. Pushes and pops add/remove blocks from the
|
2012-08-17 13:53:05 +00:00
|
|
|
* beginning/end of the list as necessary while each block tracks the used
|
|
|
|
* portion of its memory.
|
|
|
|
* One behavior to be aware of is that the pops do not immediately remove an
|
|
|
|
* empty block from the beginning/end of the list (Presumably so push/pop pairs
|
|
|
|
* on the block boundaries don't cause thrashing). This can result in the first/
|
|
|
|
* last element not residing in the first/last block.
|
|
|
|
*/
|
2011-03-15 21:27:08 +00:00
|
|
|
class SK_API SkDeque : SkNoncopyable {
|
2008-12-17 15:59:43 +00:00
|
|
|
public:
|
2012-07-16 16:58:49 +00:00
|
|
|
/**
|
|
|
|
* elemSize specifies the size of each individual element in the deque
|
|
|
|
* allocCount specifies how many elements are to be allocated as a block
|
|
|
|
*/
|
|
|
|
explicit SkDeque(size_t elemSize, int allocCount = 1);
|
|
|
|
SkDeque(size_t elemSize, void* storage, size_t storageSize, int allocCount = 1);
|
2008-12-17 15:59:43 +00:00
|
|
|
~SkDeque();
|
|
|
|
|
|
|
|
bool empty() const { return 0 == fCount; }
|
|
|
|
int count() const { return fCount; }
|
|
|
|
size_t elemSize() const { return fElemSize; }
|
|
|
|
|
2012-08-17 13:53:05 +00:00
|
|
|
const void* front() const { return fFront; }
|
|
|
|
const void* back() const { return fBack; }
|
2008-12-17 15:59:43 +00:00
|
|
|
|
|
|
|
void* front() {
|
|
|
|
return (void*)((const SkDeque*)this)->front();
|
|
|
|
}
|
|
|
|
|
|
|
|
void* back() {
|
|
|
|
return (void*)((const SkDeque*)this)->back();
|
|
|
|
}
|
|
|
|
|
2012-08-17 13:53:05 +00:00
|
|
|
/**
|
|
|
|
* push_front and push_back return a pointer to the memory space
|
|
|
|
* for the new element
|
|
|
|
*/
|
2008-12-17 15:59:43 +00:00
|
|
|
void* push_front();
|
|
|
|
void* push_back();
|
2011-02-22 13:16:38 +00:00
|
|
|
|
2008-12-17 15:59:43 +00:00
|
|
|
void pop_front();
|
|
|
|
void pop_back();
|
|
|
|
|
|
|
|
private:
|
2012-07-16 16:58:49 +00:00
|
|
|
struct Block;
|
2008-12-17 15:59:43 +00:00
|
|
|
|
|
|
|
public:
|
2012-07-16 16:58:49 +00:00
|
|
|
class Iter {
|
2008-12-17 15:59:43 +00:00
|
|
|
public:
|
2012-07-16 16:58:49 +00:00
|
|
|
enum IterStart {
|
|
|
|
kFront_IterStart,
|
2018-05-21 15:56:57 +00:00
|
|
|
kBack_IterStart,
|
2012-07-16 16:58:49 +00:00
|
|
|
};
|
|
|
|
|
2011-03-03 13:54:13 +00:00
|
|
|
/**
|
|
|
|
* Creates an uninitialized iterator. Must be reset()
|
|
|
|
*/
|
2012-07-16 16:58:49 +00:00
|
|
|
Iter();
|
2011-03-03 13:54:13 +00:00
|
|
|
|
2012-07-16 16:58:49 +00:00
|
|
|
Iter(const SkDeque& d, IterStart startLoc);
|
2008-12-17 15:59:43 +00:00
|
|
|
void* next();
|
2012-07-16 16:58:49 +00:00
|
|
|
void* prev();
|
2008-12-17 15:59:43 +00:00
|
|
|
|
2012-07-16 16:58:49 +00:00
|
|
|
void reset(const SkDeque& d, IterStart startLoc);
|
2011-03-03 13:54:13 +00:00
|
|
|
|
2008-12-17 15:59:43 +00:00
|
|
|
private:
|
2013-08-16 00:53:34 +00:00
|
|
|
SkDeque::Block* fCurBlock;
|
2008-12-17 15:59:43 +00:00
|
|
|
char* fPos;
|
|
|
|
size_t fElemSize;
|
|
|
|
};
|
2012-08-23 18:09:54 +00:00
|
|
|
|
2012-07-16 16:58:49 +00:00
|
|
|
// Inherit privately from Iter to prevent access to reverse iteration
|
|
|
|
class F2BIter : private Iter {
|
|
|
|
public:
|
|
|
|
F2BIter() {}
|
|
|
|
|
|
|
|
/**
|
2012-08-23 18:09:54 +00:00
|
|
|
* Wrap Iter's 2 parameter ctor to force initialization to the
|
2012-07-16 16:58:49 +00:00
|
|
|
* beginning of the deque
|
|
|
|
*/
|
|
|
|
F2BIter(const SkDeque& d) : INHERITED(d, kFront_IterStart) {}
|
|
|
|
|
|
|
|
using Iter::next;
|
|
|
|
|
2012-08-23 18:09:54 +00:00
|
|
|
/**
|
|
|
|
* Wrap Iter::reset to force initialization to the beginning of the
|
2012-07-16 16:58:49 +00:00
|
|
|
* deque
|
|
|
|
*/
|
|
|
|
void reset(const SkDeque& d) {
|
|
|
|
this->INHERITED::reset(d, kFront_IterStart);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
typedef Iter INHERITED;
|
|
|
|
};
|
2008-12-17 15:59:43 +00:00
|
|
|
|
|
|
|
private:
|
2012-07-16 16:58:49 +00:00
|
|
|
// allow unit test to call numBlocksAllocated
|
2012-08-23 18:09:54 +00:00
|
|
|
friend class DequeUnitTestHelper;
|
2012-07-16 16:58:49 +00:00
|
|
|
|
2012-08-17 13:53:05 +00:00
|
|
|
void* fFront;
|
|
|
|
void* fBack;
|
|
|
|
|
|
|
|
Block* fFrontBlock;
|
|
|
|
Block* fBackBlock;
|
2008-12-17 15:59:43 +00:00
|
|
|
size_t fElemSize;
|
|
|
|
void* fInitialStorage;
|
2012-07-16 16:58:49 +00:00
|
|
|
int fCount; // number of elements in the deque
|
|
|
|
int fAllocCount; // number of elements to allocate per block
|
|
|
|
|
|
|
|
Block* allocateBlock(int allocCount);
|
|
|
|
void freeBlock(Block* block);
|
2011-02-22 13:16:38 +00:00
|
|
|
|
2012-07-16 16:58:49 +00:00
|
|
|
/**
|
|
|
|
* This returns the number of chunk blocks allocated by the deque. It
|
|
|
|
* can be used to gauge the effectiveness of the selected allocCount.
|
|
|
|
*/
|
|
|
|
int numBlocksAllocated() const;
|
2008-12-17 15:59:43 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
#endif
|