Update contrib.
1 // Copyright (c) 2006-2010 Nokia Corporation and/or its subsidiary(-ies).
2 // All rights reserved.
3 // This component and the accompanying materials are made available
4 // under the terms of "Eclipse Public License v1.0"
5 // which accompanies this distribution, and is available
6 // at the URL "http://www.eclipse.org/legal/epl-v10.html".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
17 #include "SqlBufFlat.h"
18 #include "OstTraceDefinitions.h"
21 Sets the flat buffer pointer to NULL
23 RSqlBufFlat::RSqlBufFlat() :
31 "Resource acquisiton" method.
32 Sets the elements count of a new or already existing flat buffer.
34 The occupied memory won't be freed (in case of buffer resizing).
35 The buffer content is not preserved (in case of buffer resizing).
37 All elements set to have:
42 @param aCount Desired flat buffer elements count
44 @return KErrNone, The operation has completed successfully;
45 KErrNoMemory, Out of memory condition has occured.
47 TInt RSqlBufFlat::SetCount(TInt aCount)
49 __ASSERT_DEBUG(aCount >= 0, __SQLPANIC(ESqlPanicBadArgument));
50 TInt headerSize = sizeof(RSqlBufFlat::TCell) * aCount;
51 TInt newSize = headerSize + sizeof(RSqlBufFlat::TBufFlat);
52 if(DoReAlloc(newSize) != KErrNone)
56 TBufFlat& buf = *iBuf;
58 buf.iHeaderSize = headerSize;
62 SQLFLATBUF_INVARIANT();
67 Reallocates the amount of the occupied by the flat buffer memory.
68 The operation preserves the content of the flat buffer.
69 Note: if the new size is less or equal to the max size of the buffer,
70 no memory will be allocated.
72 @param aSize Desired flat buffer size in bytes
74 @return KErrNone, The operation has completed successfully;
75 KErrNoMemory, Out of memory condition has occured.
77 TInt RSqlBufFlat::ReAlloc(TInt aSize)
79 SQLFLATBUF_INVARIANT();
80 TInt err = DoReAlloc(aSize);
81 SQLFLATBUF_INVARIANT();
86 Cleans up the flat buffer content and
87 frees the occupied memory if the memory is above KBufLimit.
88 The count of elements is preserved.
90 If the buffer size is bigger than KBufLimit,
91 the buffer will be reallocated and the buffer content - not preserved.
93 If the buffer size is less or equal to KBufLimit,
94 no memory will be reallocated and the buffer preserves its content.
96 It is guaranteed that the reallocated buffer will have the same address in the heap
99 void RSqlBufFlat::ResetAndMinimize()
101 SQLFLATBUF_INVARIANT();
104 const TInt KBufLimit = Max((TInt)RSqlBufFlat::EExpandSize, SysDataSize());
105 const TBufFlat* oldBuf = iBuf;
107 const TInt KBufLimit = Max((8 * 1024), SysDataSize());
109 if(iMaxSize > KBufLimit)
111 iMaxSize = KBufLimit - 1; //to force the reallocation
112 (void)DoReAlloc(KBufLimit);//User::ReAlloc() does not fail if the new requested size is less than the current block size
114 __ASSERT_DEBUG(oldBuf == iBuf, __SQLPANIC(ESqlPanicInternalError));
115 SQLFLATBUF_INVARIANT();
119 Cleans up the flat buffer content but does not free the occupied memory.
120 The count of elements is preserved.
122 All elements set to have:
125 - zero data position;
127 The element count is preserved.
129 void RSqlBufFlat::Reset()
131 SQLFLATBUF_INVARIANT();
132 iBuf->iSize = SysDataSize();
134 SQLFLATBUF_INVARIANT();
138 Closes the flat bufer and frees the allocated memory.
140 void RSqlBufFlat::Close()
147 Sets the content of a field.
149 @param aIndex Field index
150 @param aType Field type
151 @param aData Field data, may be NULL
152 @param aDataLength Field data length, may be 0
154 @return KErrNone, The operation has completed successfully;
155 KErrNoMemory, Out of memory condition has occured.
157 TInt RSqlBufFlat::SetField(TInt aIndex, TInt aType, const void* aData, TInt aDataLength)
159 SQLFLATBUF_INVARIANT();
160 __ASSERT_DEBUG((TUint)aIndex < iBuf->iCount, __SQLPANIC(ESqlPanicBadArgument));
161 __ASSERT_DEBUG((TUint)aType < RSqlBufFlat::EMaxType, __SQLPANIC(ESqlPanicBadArgument));
162 __ASSERT_DEBUG((TUint)aDataLength < RSqlBufFlat::EMaxLength, __SQLPANIC(ESqlPanicBadArgument));
163 if(aData) //field value "present"
165 __ASSERT_DEBUG(aDataLength >= 0, __SQLPANIC(ESqlPanicBadArgument));
168 if(Reserve(aDataLength) != KErrNone)
173 DoSet(aIndex, aType, aData, aDataLength);
175 else if(aDataLength == 0) //data is NULL, length is 0 - "null" field
177 DoSet(aIndex, aType, NULL, 0);
179 else //field value "not present"
181 RSqlBufFlat::TCell& cell = *(Header() + aIndex);
182 cell.iBits = (TUint)(((TUint)aType << RSqlBufFlat::EWidthLen) | (TUint)aDataLength);
185 SQLFLATBUF_INVARIANT();
190 Initialzies the flat buffer header.
196 void RSqlBufFlat::DoInit()
198 TBufFlat& buf = *iBuf;
199 __ASSERT_DEBUG(buf.iCount >= 0, __SQLPANIC(ESqlPanicInternalError));
200 __ASSERT_DEBUG(buf.iSize <= iMaxSize, __SQLPANIC(ESqlPanicInternalError));
201 __ASSERT_DEBUG(buf.iHeaderSize == sizeof(RSqlBufFlat::TCell) * buf.iCount, __SQLPANIC(ESqlPanicInternalError));
202 if(buf.iHeaderSize > 0)
204 Mem::FillZ(Header(), buf.iHeaderSize);
209 Reallocates the amount of the occupied by the flat buffer memory
210 (only in case the requested size is bigger than the buffer size or the buffer does not exist).
211 The operation preserves the content of the flat buffer.
213 @param aSize Desired flat buffer size in bytes.
215 @return KErrNone, The operation has completed successfully;
216 KErrNoMemory, Out of memory condition has occured.
218 TInt RSqlBufFlat::DoReAlloc(TInt aSize)
220 if(!iBuf || iMaxSize < aSize)
222 //Calculate buffer new size (sometimes allocates more, for example, if
223 //aSize % RSqlBufFlat::EExpandSize == 0, then one more RSqlBufFlat::EExpandSize page is allocated).
224 TInt newSize = (aSize / RSqlBufFlat::EExpandSize + 1) * RSqlBufFlat::EExpandSize;
225 RSqlBufFlat::TBufFlat* newBuf = static_cast <RSqlBufFlat::TBufFlat*> (User::ReAlloc(iBuf, newSize));
237 Initialzes a flat buffer field.
238 A memory for the field data has to be allocated before the call.
240 @param aIndex Field index
241 @param aType Field type
242 @param aData Field data, may be NULL
243 @param aDataLength Field data length, may be 0
245 void RSqlBufFlat::DoSet(TInt aIndex, TInt aType, const void* aData, TInt aDataLength)
247 TBufFlat& buf = *iBuf;
248 __ASSERT_DEBUG((TUint)aDataLength < RSqlBufFlat::EMaxLength, __SQLPANIC(ESqlPanicBadArgument));
249 __ASSERT_DEBUG(aDataLength > 0 ? aData != NULL : ETrue, __SQLPANIC(ESqlPanicBadArgument));
250 __ASSERT_DEBUG(aDataLength <= (iMaxSize - buf.iSize), __SQLPANIC(ESqlPanicInternalError));
251 __ASSERT_DEBUG(::IsAligned8(buf.iSize), __SQLPANIC(ESqlPanicInternalError));
252 RSqlBufFlat::TCell& cell = *(Header() + aIndex);
253 cell.iBits = (TUint)(((TUint)aType << RSqlBufFlat::EWidthLen) | (TUint)aDataLength);
254 cell.iPos = 1; //not 0, because 0 means "not present"
255 if(aDataLength > 0) //for fields with length > 0 set the data and reinitalize cell.iPos
258 Mem::Copy(reinterpret_cast <TUint8*> (iBuf) + buf.iSize, &KSqlBufFlatMagicValue, sizeof(KSqlBufFlatMagicValue));
259 buf.iSize += sizeof(KSqlBufFlatMagicValue);
261 cell.iPos = buf.iSize - sizeof(RSqlBufFlat::TBufFlat);
262 Mem::Copy(reinterpret_cast <TUint8*> (iBuf) + buf.iSize, reinterpret_cast <const TUint8*> (aData), aDataLength);
263 buf.iSize += ::AlignedLen8(aDataLength); //align the next field start position
264 //it is guaranteed that this "+" operation will not make iSize bigger than
265 //iMaxSize, because the memory allocations are 8-byte aligned
266 //(even RSqlBufFlat::EExpandSize aligned)
271 Makes sure that the flat buffer has enough free space for a block of data with "aLength" length.
272 The function may reallocated the buffer if there is not enough space.
274 @param aLength The requested free memory length.
276 @return KErrNone, The operation has completed successfully;
277 KErrNoMemory, Out of memory condition has occured.
279 TInt RSqlBufFlat::Reserve(TInt aLength)
282 TInt diff = aLength + sizeof(KSqlBufFlatMagicValue) - Available();
284 TInt diff = aLength - Available();
286 return diff > 0 ? DoReAlloc(iMaxSize + diff) : KErrNone;
291 Panics in _DEBUG mode if the flat buffer content is inconsistent.
293 void RSqlBufFlat::Invariant() const
295 __ASSERT_DEBUG(iBuf != NULL, __SQLPANIC(ESqlPanicInternalError));
296 const TBufFlat& buf = *iBuf;
297 __ASSERT_DEBUG(buf.iCount >= 0, __SQLPANIC(ESqlPanicInternalError));
298 __ASSERT_DEBUG(buf.iHeaderSize == sizeof(RSqlBufFlat::TCell) * buf.iCount, __SQLPANIC(ESqlPanicInternalError));
299 __ASSERT_DEBUG(::IsAligned8(buf.iSize), __SQLPANIC(ESqlPanicInternalError));
300 __ASSERT_DEBUG(buf.iSize >= buf.iHeaderSize + sizeof(RSqlBufFlat::TBufFlat), __SQLPANIC(ESqlPanicInternalError));
301 __ASSERT_DEBUG(buf.iSize <= iMaxSize, __SQLPANIC(ESqlPanicInternalError));
302 __ASSERT_DEBUG(buf.iSize <= User::AllocLen(iBuf), __SQLPANIC(ESqlPanicInternalError));
303 for(TInt i=0;i<(TInt)buf.iCount;++i)
305 const RSqlBufFlat::TCell& cell = *((reinterpret_cast <const RSqlBufFlat::TCell*> (iBuf + 1)) + i);
306 __ASSERT_DEBUG(cell.Type() < RSqlBufFlat::EMaxType, __SQLPANIC(ESqlPanicInternalError));
307 if(cell.Size() > 0 && cell.iPos >= buf.iHeaderSize) //only for present fields with length > 0
309 __ASSERT_DEBUG((TUint)cell.Size() <= buf.iSize, __SQLPANIC(ESqlPanicInternalError));
310 __ASSERT_DEBUG(cell.iPos < (buf.iSize - sizeof(RSqlBufFlat::TBufFlat)), __SQLPANIC(ESqlPanicInternalError));
311 TUint64 val = *(TUint64*)(reinterpret_cast <TUint8*> (iBuf) + cell.iPos + sizeof(RSqlBufFlat::TBufFlat) - sizeof(KSqlBufFlatMagicValue));
312 __ASSERT_DEBUG(val == KSqlBufFlatMagicValue, __SQLPANIC(ESqlPanicInternalError));