os/persistentdata/persistentstorage/sql/SRC/Common/SqlBufFlat.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 //
    15 
    16 #include "SqlUtil.h"
    17 #include "SqlBufFlat.h"
    18 #include "OstTraceDefinitions.h"
    19 
    20 /**
    21 Sets the flat buffer pointer to NULL
    22 */
    23 RSqlBufFlat::RSqlBufFlat() :
    24 	iBuf(NULL),
    25 	iMaxSize(0),
    26 	iBufPtr(NULL, 0)
    27 	{
    28 	}
    29 
    30 /**
    31 "Resource acquisiton" method.
    32 Sets the elements count of a new or already existing flat buffer.
    33 
    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).
    36 
    37 All elements set to have:
    38 - invalid type;
    39 - zero length;
    40 - zero data position;
    41 
    42 @param aCount Desired flat buffer elements count
    43 
    44 @return KErrNone, The operation has completed successfully;
    45 		KErrNoMemory, Out of memory condition has occured.
    46 */
    47 TInt RSqlBufFlat::SetCount(TInt aCount)
    48 	{
    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)
    53 		{
    54 		return KErrNoMemory;	
    55 		}
    56 	TBufFlat& buf = *iBuf;
    57 	buf.iCount = aCount;
    58 	buf.iHeaderSize = headerSize;
    59 	buf.iSize = newSize;
    60 	buf.iReserved = 0;
    61 	DoInit();
    62 	SQLFLATBUF_INVARIANT();
    63 	return KErrNone;	
    64 	}
    65 
    66 /**
    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.
    71 
    72 @param aSize Desired flat buffer size in bytes
    73 
    74 @return KErrNone, The operation has completed successfully;
    75 		KErrNoMemory, Out of memory condition has occured.
    76 */
    77 TInt RSqlBufFlat::ReAlloc(TInt aSize)
    78 	{
    79 	SQLFLATBUF_INVARIANT();
    80 	TInt err = DoReAlloc(aSize);
    81 	SQLFLATBUF_INVARIANT();
    82 	return err;
    83 	}
    84 
    85 /**
    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.
    89 
    90 If the buffer size is bigger than KBufLimit,
    91 the buffer will be reallocated and the buffer content - not preserved.
    92 
    93 If the buffer size is less or equal to KBufLimit,
    94 no memory will be reallocated and the buffer preserves its content.
    95 
    96 It is guaranteed that the reallocated buffer will have the same address in the heap 
    97 as the original one.
    98 */	
    99 void RSqlBufFlat::ResetAndMinimize()
   100 	{
   101 	SQLFLATBUF_INVARIANT();
   102 	Reset();
   103 #ifdef _DEBUG	
   104 	const TInt KBufLimit = Max((TInt)RSqlBufFlat::EExpandSize, SysDataSize());
   105 	const TBufFlat* oldBuf = iBuf;
   106 #else
   107 	const TInt KBufLimit = Max((8 * 1024), SysDataSize());
   108 #endif
   109 	if(iMaxSize > KBufLimit)
   110 		{
   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
   113 		}
   114 	__ASSERT_DEBUG(oldBuf == iBuf, __SQLPANIC(ESqlPanicInternalError));
   115 	SQLFLATBUF_INVARIANT();
   116 	}
   117 
   118 /**
   119 Cleans up the flat buffer content but does not free the occupied memory.
   120 The count of elements is preserved.
   121 
   122 All elements set to have:
   123 - invalid type;
   124 - zero length;
   125 - zero data position;
   126 
   127 The element count is preserved.
   128 */
   129 void RSqlBufFlat::Reset()
   130 	{
   131 	SQLFLATBUF_INVARIANT();
   132 	iBuf->iSize = SysDataSize();
   133 	DoInit();
   134 	SQLFLATBUF_INVARIANT();
   135 	}
   136 
   137 /**
   138 Closes the flat bufer and frees the allocated memory.
   139 */
   140 void RSqlBufFlat::Close()
   141 	{
   142 	User::Free(iBuf);
   143 	iBuf = NULL;	
   144 	}
   145 
   146 /**
   147 Sets the content of a field.
   148 
   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
   153 
   154 @return KErrNone, The operation has completed successfully;
   155 		KErrNoMemory, Out of memory condition has occured.
   156 */
   157 TInt RSqlBufFlat::SetField(TInt aIndex, TInt aType, const void* aData, TInt aDataLength)
   158 	{
   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"
   164 		{
   165 		__ASSERT_DEBUG(aDataLength >= 0, __SQLPANIC(ESqlPanicBadArgument));
   166 		if(aDataLength > 0)
   167 			{
   168 			if(Reserve(aDataLength) != KErrNone)
   169 				{
   170 				return KErrNoMemory;	
   171 				}
   172 			}
   173 		DoSet(aIndex, aType, aData, aDataLength);
   174 		}
   175 	else if(aDataLength == 0)		//data is NULL, length is 0 - "null" field
   176 		{
   177 		DoSet(aIndex, aType, NULL, 0);
   178 		}
   179 	else							//field value "not present"
   180 		{
   181 		RSqlBufFlat::TCell& cell = *(Header() + aIndex);
   182 		cell.iBits = (TUint)(((TUint)aType << RSqlBufFlat::EWidthLen) | (TUint)aDataLength);
   183 		cell.iPos = 0;
   184 		}
   185 	SQLFLATBUF_INVARIANT();
   186 	return KErrNone;
   187 	}
   188 	
   189 /**
   190 Initialzies the flat buffer header.
   191 All field set:
   192 - invalid type;
   193 - zero length;
   194 - "Not present";
   195 */	
   196 void RSqlBufFlat::DoInit()
   197 	{
   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)
   203 		{
   204 		Mem::FillZ(Header(), buf.iHeaderSize);
   205 		}
   206 	}
   207 
   208 /**
   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.
   212 
   213 @param aSize Desired flat buffer size in bytes.
   214 
   215 @return KErrNone, The operation has completed successfully;
   216 		KErrNoMemory, Out of memory condition has occured.
   217 */
   218 TInt RSqlBufFlat::DoReAlloc(TInt aSize)
   219 	{
   220 	if(!iBuf || iMaxSize < aSize)
   221 		{
   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));
   226 		if(!newBuf)
   227 			{
   228 			return KErrNoMemory;	
   229 			}
   230 		iBuf = newBuf;
   231 		iMaxSize = newSize;
   232 		}
   233 	return KErrNone;
   234 	}
   235 
   236 /**
   237 Initialzes a flat buffer field.
   238 A memory for the field data has to be allocated before the call.
   239 
   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
   244 */
   245 void RSqlBufFlat::DoSet(TInt aIndex, TInt aType, const void* aData, TInt aDataLength)
   246 	{
   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
   256 		{
   257 #ifdef _DEBUG
   258 		Mem::Copy(reinterpret_cast <TUint8*> (iBuf) + buf.iSize, &KSqlBufFlatMagicValue, sizeof(KSqlBufFlatMagicValue));
   259 		buf.iSize += sizeof(KSqlBufFlatMagicValue);
   260 #endif		
   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)
   267 		}
   268 	}
   269 
   270 /**
   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.
   273 
   274 @param aLength The requested free memory length.
   275 
   276 @return KErrNone, The operation has completed successfully;
   277 		KErrNoMemory, Out of memory condition has occured.
   278 */
   279 TInt RSqlBufFlat::Reserve(TInt aLength)
   280 	{
   281 #ifdef _DEBUG
   282 	TInt diff = aLength + sizeof(KSqlBufFlatMagicValue) - Available();
   283 #else
   284 	TInt diff = aLength - Available();
   285 #endif		
   286 	return diff > 0 ? DoReAlloc(iMaxSize + diff) : KErrNone;
   287 	}
   288 
   289 #ifdef _DEBUG
   290 /**
   291 Panics in _DEBUG mode if the flat buffer content is inconsistent.
   292 */
   293 void RSqlBufFlat::Invariant() const
   294 	{
   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)
   304 		{
   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
   308 			{
   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));
   313 			}
   314 		}
   315 	}
   316 #endif//_DEBUG