| // Copyright 2014 PDFium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com |
| |
| #ifndef XFA_FGAS_CRT_FGAS_UTILS_H_ |
| #define XFA_FGAS_CRT_FGAS_UTILS_H_ |
| |
| #include "core/fxcrt/include/fx_coordinates.h" |
| #include "xfa/fgas/crt/fgas_memory.h" |
| |
| class FX_BASEARRAYDATA; |
| |
| class CFX_BaseArray : public CFX_Target { |
| protected: |
| CFX_BaseArray(int32_t iGrowSize, int32_t iBlockSize); |
| ~CFX_BaseArray(); |
| int32_t GetSize() const; |
| int32_t GetBlockSize() const; |
| uint8_t* AddSpaceTo(int32_t index); |
| uint8_t* GetAt(int32_t index) const; |
| uint8_t* GetBuffer() const; |
| int32_t Append(const CFX_BaseArray& src, |
| int32_t iStart = 0, |
| int32_t iCount = -1); |
| int32_t Copy(const CFX_BaseArray& src, |
| int32_t iStart = 0, |
| int32_t iCount = -1); |
| int32_t RemoveLast(int32_t iCount = -1); |
| void RemoveAll(FX_BOOL bLeaveMemory = FALSE); |
| |
| FX_BASEARRAYDATA* m_pData; |
| }; |
| |
| template <class baseType> |
| class CFX_BaseArrayTemplate : public CFX_BaseArray { |
| public: |
| CFX_BaseArrayTemplate(int32_t iGrowSize = 100) |
| : CFX_BaseArray(iGrowSize, sizeof(baseType)) {} |
| CFX_BaseArrayTemplate(int32_t iGrowSize, int32_t iBlockSize) |
| : CFX_BaseArray(iGrowSize, iBlockSize) {} |
| int32_t GetSize() const { return CFX_BaseArray::GetSize(); } |
| int32_t GetBlockSize() const { return CFX_BaseArray::GetBlockSize(); } |
| baseType* AddSpace() { |
| return (baseType*)CFX_BaseArray::AddSpaceTo(CFX_BaseArray::GetSize()); |
| } |
| int32_t Add(const baseType& element) { |
| int32_t index = CFX_BaseArray::GetSize(); |
| *(baseType*)CFX_BaseArray::AddSpaceTo(index) = element; |
| return index; |
| } |
| baseType* GetBuffer() const { return (baseType*)CFX_BaseArray::GetBuffer(); } |
| baseType& GetAt(int32_t index) const { |
| return *(baseType*)CFX_BaseArray::GetAt(index); |
| } |
| baseType* GetPtrAt(int32_t index) const { |
| return (baseType*)CFX_BaseArray::GetAt(index); |
| } |
| void SetAt(int32_t index, const baseType& element) { |
| *(baseType*)CFX_BaseArray::GetAt(index) = element; |
| } |
| void SetAtGrow(int32_t index, const baseType& element) { |
| *(baseType*)CFX_BaseArray::AddSpaceTo(index) = element; |
| } |
| int32_t Append(const CFX_BaseArrayTemplate& src, |
| int32_t iStart = 0, |
| int32_t iCount = -1) { |
| return CFX_BaseArray::Append(src, iStart, iCount); |
| } |
| int32_t Copy(const CFX_BaseArrayTemplate& src, |
| int32_t iStart = 0, |
| int32_t iCount = -1) { |
| return CFX_BaseArray::Copy(src, iStart, iCount); |
| } |
| int32_t RemoveLast(int32_t iCount = -1) { |
| return CFX_BaseArray::RemoveLast(iCount); |
| } |
| void RemoveAll(FX_BOOL bLeaveMemory = FALSE) { |
| CFX_BaseArray::RemoveAll(bLeaveMemory); |
| } |
| }; |
| |
| template <class baseType> |
| class CFX_ObjectBaseArrayTemplate : public CFX_BaseArray { |
| public: |
| CFX_ObjectBaseArrayTemplate(int32_t iGrowSize = 100) |
| : CFX_BaseArray(iGrowSize, sizeof(baseType)) {} |
| ~CFX_ObjectBaseArrayTemplate() { RemoveAll(FALSE); } |
| int32_t GetSize() const { return CFX_BaseArray::GetSize(); } |
| int32_t GetBlockSize() const { return CFX_BaseArray::GetBlockSize(); } |
| int32_t Add(const baseType& element) { |
| int32_t index = CFX_BaseArray::GetSize(); |
| baseType* p = (baseType*)CFX_BaseArray::AddSpaceTo(index); |
| new ((void*)p) baseType(element); |
| return index; |
| } |
| baseType& GetAt(int32_t index) const { |
| return *(baseType*)CFX_BaseArray::GetAt(index); |
| } |
| baseType* GetPtrAt(int32_t index) const { |
| return (baseType*)CFX_BaseArray::GetAt(index); |
| } |
| int32_t Append(const CFX_ObjectBaseArrayTemplate& src, |
| int32_t iStart = 0, |
| int32_t iCount = -1) { |
| ASSERT(GetBlockSize() == src.GetBlockSize()); |
| if (iCount == 0) { |
| return 0; |
| } |
| int32_t iSize = src.GetSize(); |
| ASSERT(iStart > -1 && iStart < iSize); |
| if (iCount < 0) { |
| iCount = iSize; |
| } |
| if (iStart + iCount > iSize) { |
| iCount = iSize - iStart; |
| } |
| if (iCount < 1) { |
| return 0; |
| } |
| iSize = CFX_BaseArray::GetSize(); |
| CFX_BaseArray::AddSpaceTo(iSize + iCount - 1); |
| uint8_t** pStart = CFX_BaseArray::GetAt(iSize); |
| int32_t iBlockSize = CFX_BaseArray::GetBlockSize(); |
| iSize = iStart + iCount; |
| for (int32_t i = iStart; i < iSize; i++) { |
| FXTARGET_NewWith((void*)pStart) baseType(src.GetAt(i)); |
| pStart += iBlockSize; |
| } |
| return iCount; |
| } |
| int32_t Copy(const CFX_ObjectBaseArrayTemplate& src, |
| int32_t iStart = 0, |
| int32_t iCount = -1) { |
| ASSERT(GetBlockSize() == src.GetBlockSize()); |
| if (iCount == 0) { |
| return 0; |
| } |
| int32_t iSize = src.GetSize(); |
| ASSERT(iStart > -1 && iStart < iSize); |
| if (iCount < 0) { |
| iCount = iSize; |
| } |
| if (iStart + iCount > iSize) { |
| iCount = iSize - iStart; |
| } |
| if (iCount < 1) { |
| return 0; |
| } |
| RemoveAll(TRUE); |
| CFX_BaseArray::AddSpaceTo(iCount - 1); |
| uint8_t** pStart = CFX_BaseArray::GetAt(0); |
| int32_t iBlockSize = CFX_BaseArray::GetBlockSize(); |
| iSize = iStart + iCount; |
| for (int32_t i = iStart; i < iSize; i++) { |
| new ((void*)pStart) baseType(src.GetAt(i)); |
| pStart += iBlockSize; |
| } |
| return iCount; |
| } |
| int32_t RemoveLast(int32_t iCount = -1) { |
| int32_t iSize = CFX_BaseArray::GetSize(); |
| if (iCount < 0 || iCount > iSize) { |
| iCount = iSize; |
| } |
| if (iCount == 0) { |
| return iSize; |
| } |
| for (int32_t i = iSize - iCount; i < iSize; i++) { |
| ((baseType*)GetPtrAt(i))->~baseType(); |
| } |
| return CFX_BaseArray::RemoveLast(iCount); |
| } |
| void RemoveAll(FX_BOOL bLeaveMemory = FALSE) { |
| int32_t iSize = CFX_BaseArray::GetSize(); |
| for (int32_t i = 0; i < iSize; i++) { |
| ((baseType*)GetPtrAt(i))->~baseType(); |
| } |
| CFX_BaseArray::RemoveAll(bLeaveMemory); |
| } |
| }; |
| |
| class CFX_BaseMassArrayImp : public CFX_Target { |
| public: |
| CFX_BaseMassArrayImp(int32_t iChunkSize, int32_t iBlockSize); |
| ~CFX_BaseMassArrayImp(); |
| uint8_t* AddSpace() { return AddSpaceTo(m_iBlockCount); } |
| uint8_t* AddSpaceTo(int32_t index); |
| uint8_t* GetAt(int32_t index) const; |
| int32_t Append(const CFX_BaseMassArrayImp& src, |
| int32_t iStart = 0, |
| int32_t iCount = -1); |
| int32_t Copy(const CFX_BaseMassArrayImp& src, |
| int32_t iStart = 0, |
| int32_t iCount = -1); |
| int32_t RemoveLast(int32_t iCount = -1); |
| void RemoveAll(FX_BOOL bLeaveMemory = FALSE); |
| |
| int32_t m_iChunkSize; |
| int32_t m_iBlockSize; |
| int32_t m_iChunkCount; |
| int32_t m_iBlockCount; |
| CFX_ArrayTemplate<void*>* m_pData; |
| |
| protected: |
| void Append(int32_t iDstStart, |
| const CFX_BaseMassArrayImp& src, |
| int32_t iSrcStart = 0, |
| int32_t iSrcCount = -1); |
| }; |
| |
| class CFX_BaseMassArray : public CFX_Target { |
| protected: |
| CFX_BaseMassArray(int32_t iChunkSize, int32_t iBlockSize); |
| ~CFX_BaseMassArray(); |
| int32_t GetSize() const; |
| uint8_t* AddSpaceTo(int32_t index); |
| uint8_t* GetAt(int32_t index) const; |
| int32_t Append(const CFX_BaseMassArray& src, |
| int32_t iStart = 0, |
| int32_t iCount = -1); |
| int32_t Copy(const CFX_BaseMassArray& src, |
| int32_t iStart = 0, |
| int32_t iCount = -1); |
| int32_t RemoveLast(int32_t iCount = -1); |
| void RemoveAll(FX_BOOL bLeaveMemory = FALSE); |
| CFX_BaseMassArrayImp* m_pData; |
| }; |
| |
| template <class baseType> |
| class CFX_MassArrayTemplate : public CFX_BaseMassArray { |
| public: |
| CFX_MassArrayTemplate(int32_t iChunkSize = 100) |
| : CFX_BaseMassArray(iChunkSize, sizeof(baseType)) {} |
| CFX_MassArrayTemplate(int32_t iChunkSize, int32_t iBlockSize) |
| : CFX_BaseMassArray(iChunkSize, iBlockSize) {} |
| int32_t GetSize() const { return CFX_BaseMassArray::GetSize(); } |
| baseType* AddSpace() { |
| return (baseType*)CFX_BaseMassArray::AddSpaceTo( |
| CFX_BaseMassArray::GetSize()); |
| } |
| int32_t Add(const baseType& element) { |
| int32_t index = CFX_BaseMassArray::GetSize(); |
| *(baseType*)CFX_BaseMassArray::AddSpaceTo(index) = element; |
| return index; |
| } |
| baseType& GetAt(int32_t index) const { |
| return *(baseType*)CFX_BaseMassArray::GetAt(index); |
| } |
| baseType* GetPtrAt(int32_t index) const { |
| return (baseType*)CFX_BaseMassArray::GetAt(index); |
| } |
| void SetAt(int32_t index, const baseType& element) { |
| *(baseType*)CFX_BaseMassArray::GetAt(index) = element; |
| } |
| void SetAtGrow(int32_t index, const baseType& element) { |
| *(baseType*)CFX_BaseMassArray::AddSpaceTo(index) = element; |
| } |
| int32_t Append(const CFX_MassArrayTemplate& src, |
| int32_t iStart = 0, |
| int32_t iCount = -1) { |
| return CFX_BaseMassArray::Append(src, iStart, iCount); |
| } |
| int32_t Copy(const CFX_MassArrayTemplate& src, |
| int32_t iStart = 0, |
| int32_t iCount = -1) { |
| return CFX_BaseMassArray::Copy(src, iStart, iCount); |
| } |
| int32_t RemoveLast(int32_t iCount = -1) { |
| return CFX_BaseMassArray::RemoveLast(iCount); |
| } |
| void RemoveAll(FX_BOOL bLeaveMemory = FALSE) { |
| CFX_BaseMassArray::RemoveAll(bLeaveMemory); |
| } |
| }; |
| |
| template <class baseType> |
| class CFX_ObjectMassArrayTemplate : public CFX_BaseMassArray { |
| public: |
| CFX_ObjectMassArrayTemplate(int32_t iChunkSize = 100) |
| : CFX_BaseMassArray(iChunkSize, sizeof(baseType)) {} |
| ~CFX_ObjectMassArrayTemplate() { RemoveAll(FALSE); } |
| int32_t GetSize() const { return CFX_BaseMassArray::GetSize(); } |
| int32_t Add(const baseType& element) { |
| int32_t index = CFX_BaseMassArray::GetSize(); |
| baseType* p = (baseType*)CFX_BaseMassArray::AddSpaceTo(index); |
| new ((void*)p) baseType(element); |
| return index; |
| } |
| baseType& GetAt(int32_t index) const { |
| return *(baseType*)CFX_BaseMassArray::GetAt(index); |
| } |
| baseType* GetPtrAt(int32_t index) const { |
| return (baseType*)CFX_BaseMassArray::GetAt(index); |
| } |
| int32_t Append(const CFX_ObjectMassArrayTemplate& src, |
| int32_t iStart = 0, |
| int32_t iCount = -1) { |
| if (iCount == 0) { |
| return CFX_BaseMassArray::GetSize(); |
| } |
| int32_t iSize = src.GetSize(); |
| ASSERT(iStart > -1 && iStart < iSize); |
| if (iCount < 0) { |
| iCount = iSize; |
| } |
| int32_t iEnd = iStart + iCount; |
| if (iEnd > iSize) { |
| iEnd = iSize; |
| } |
| for (int32_t i = iStart; i < iEnd; i++) { |
| Add(src.GetAt(i)); |
| } |
| return CFX_BaseMassArray::GetSize(); |
| } |
| int32_t Copy(const CFX_ObjectMassArrayTemplate& src, |
| int32_t iStart = 0, |
| int32_t iCount = -1) { |
| if (iCount == 0) { |
| return CFX_BaseMassArray::GetSize(); |
| } |
| int32_t iSize = src.GetSize(); |
| ASSERT(iStart > -1 && iStart < iSize); |
| if (iCount < 0) { |
| iCount = iSize; |
| } |
| int32_t iEnd = iStart + iCount; |
| if (iEnd > iSize) { |
| iEnd = iSize; |
| } |
| RemoveAll(TRUE); |
| for (int32_t i = iStart; i < iEnd; i++) { |
| Add(src.GetAt(i)); |
| } |
| return CFX_BaseMassArray::GetSize(); |
| } |
| int32_t RemoveLast(int32_t iCount = -1) { |
| int32_t iSize = CFX_BaseMassArray::GetSize(); |
| if (iCount < 0 || iCount > iSize) { |
| iCount = iSize; |
| } |
| if (iCount == 0) { |
| return iSize; |
| } |
| for (int32_t i = iSize - iCount; i < iSize; i++) { |
| ((baseType*)GetPtrAt(i))->~baseType(); |
| } |
| return CFX_BaseMassArray::RemoveLast(iCount); |
| } |
| void RemoveAll(FX_BOOL bLeaveMemory = FALSE) { |
| int32_t iSize = CFX_BaseMassArray::GetSize(); |
| for (int32_t i = 0; i < iSize; i++) { |
| ((baseType*)GetPtrAt(i))->~baseType(); |
| } |
| CFX_BaseMassArray::RemoveAll(bLeaveMemory); |
| } |
| }; |
| |
| class CFX_BaseDiscreteArray : public CFX_Target { |
| protected: |
| CFX_BaseDiscreteArray(int32_t iChunkSize, int32_t iBlockSize); |
| ~CFX_BaseDiscreteArray(); |
| uint8_t* AddSpaceTo(int32_t index); |
| uint8_t* GetAt(int32_t index) const; |
| void RemoveAll(); |
| void* m_pData; |
| }; |
| |
| template <class baseType> |
| class CFX_DiscreteArrayTemplate : public CFX_BaseDiscreteArray { |
| public: |
| CFX_DiscreteArrayTemplate(int32_t iChunkSize = 100) |
| : CFX_BaseDiscreteArray(iChunkSize, sizeof(baseType)) {} |
| baseType& GetAt(int32_t index, const baseType& defValue) const { |
| baseType* p = (baseType*)CFX_BaseDiscreteArray::GetAt(index); |
| return p == NULL ? (baseType&)defValue : *p; |
| } |
| baseType* GetPtrAt(int32_t index) const { |
| return (baseType*)CFX_BaseDiscreteArray::GetAt(index); |
| } |
| void SetAtGrow(int32_t index, const baseType& element) { |
| *(baseType*)CFX_BaseDiscreteArray::AddSpaceTo(index) = element; |
| } |
| void RemoveAll() { CFX_BaseDiscreteArray::RemoveAll(); } |
| }; |
| typedef CFX_DiscreteArrayTemplate<void*> CFX_PtrDiscreteArray; |
| typedef CFX_DiscreteArrayTemplate<uint32_t> CFX_DWordDiscreteArray; |
| typedef CFX_DiscreteArrayTemplate<uint16_t> CFX_WordDiscreteArray; |
| |
| class CFX_BaseStack : public CFX_Target { |
| protected: |
| CFX_BaseStack(int32_t iChunkSize, int32_t iBlockSize); |
| ~CFX_BaseStack(); |
| uint8_t* Push(); |
| void Pop(); |
| uint8_t* GetTopElement() const; |
| int32_t GetSize() const; |
| uint8_t* GetAt(int32_t index) const; |
| void RemoveAll(FX_BOOL bLeaveMemory = FALSE); |
| CFX_BaseMassArrayImp* m_pData; |
| }; |
| |
| template <class baseType> |
| class CFX_StackTemplate : public CFX_BaseStack { |
| public: |
| CFX_StackTemplate(int32_t iChunkSize = 100) |
| : CFX_BaseStack(iChunkSize, sizeof(baseType)) {} |
| int32_t Push(const baseType& element) { |
| int32_t index = CFX_BaseStack::GetSize(); |
| *(baseType*)CFX_BaseStack::Push() = element; |
| return index; |
| } |
| void Pop() { CFX_BaseStack::Pop(); } |
| baseType* GetTopElement() const { |
| return (baseType*)CFX_BaseStack::GetTopElement(); |
| } |
| int32_t GetSize() const { return CFX_BaseStack::GetSize(); } |
| baseType* GetAt(int32_t index) const { |
| return (baseType*)CFX_BaseStack::GetAt(index); |
| } |
| void RemoveAll(FX_BOOL bLeaveMemory = FALSE) { |
| CFX_BaseStack::RemoveAll(bLeaveMemory); |
| } |
| }; |
| typedef CFX_StackTemplate<void*> CFX_PtrStack; |
| typedef CFX_StackTemplate<uint32_t> CFX_DWordStack; |
| typedef CFX_StackTemplate<uint16_t> CFX_WordStack; |
| typedef CFX_StackTemplate<int32_t> CFX_Int32Stack; |
| |
| template <class baseType> |
| class CFX_ObjectStackTemplate : public CFX_BaseStack { |
| public: |
| CFX_ObjectStackTemplate(int32_t iChunkSize = 100) |
| : CFX_BaseStack(iChunkSize, sizeof(baseType)) {} |
| ~CFX_ObjectStackTemplate() { RemoveAll(); } |
| int32_t Push(const baseType& element) { |
| int32_t index = CFX_BaseStack::GetSize(); |
| baseType* p = (baseType*)CFX_BaseStack::Push(); |
| new ((void*)p) baseType(element); |
| return index; |
| } |
| void Pop() { |
| baseType* p = (baseType*)CFX_BaseStack::GetTopElement(); |
| if (p != NULL) { |
| p->~baseType(); |
| } |
| CFX_BaseStack::Pop(); |
| } |
| baseType* GetTopElement() const { |
| return (baseType*)CFX_BaseStack::GetTopElement(); |
| } |
| int32_t GetSize() const { return CFX_BaseStack::GetSize(); } |
| baseType* GetAt(int32_t index) const { |
| return (baseType*)CFX_BaseStack::GetAt(index); |
| } |
| void RemoveAll(FX_BOOL bLeaveMemory = FALSE) { |
| int32_t iSize = CFX_BaseStack::GetSize(); |
| for (int32_t i = 0; i < iSize; i++) { |
| ((baseType*)CFX_BaseStack::GetAt(i))->~baseType(); |
| } |
| CFX_BaseStack::RemoveAll(bLeaveMemory); |
| } |
| int32_t Copy(const CFX_ObjectStackTemplate& src, |
| int32_t iStart = 0, |
| int32_t iCount = -1) { |
| if (iCount == 0) { |
| return CFX_BaseStack::GetSize(); |
| } |
| int32_t iSize = src.GetSize(); |
| ASSERT(iStart > -1 && iStart < iSize); |
| if (iCount < 0) { |
| iCount = iSize; |
| } |
| int32_t iEnd = iStart + iCount; |
| if (iEnd > iSize) { |
| iEnd = iSize; |
| } |
| RemoveAll(TRUE); |
| for (int32_t i = iStart; i < iEnd; i++) { |
| Push(*src.GetAt(i)); |
| } |
| return CFX_BaseStack::GetSize(); |
| } |
| }; |
| |
| #endif // XFA_FGAS_CRT_FGAS_UTILS_H_ |