os/persistentdata/persistentstorage/sql/SRC/Common/SqlBufFlat.cpp
author sl@SLION-WIN7.fritz.box
Fri, 15 Jun 2012 03:10:57 +0200
changeset 0 bde4ae8d615e
permissions -rw-r--r--
First public contribution.
sl@0
     1
// Copyright (c) 2006-2010 Nokia Corporation and/or its subsidiary(-ies).
sl@0
     2
// All rights reserved.
sl@0
     3
// This component and the accompanying materials are made available
sl@0
     4
// under the terms of "Eclipse Public License v1.0"
sl@0
     5
// which accompanies this distribution, and is available
sl@0
     6
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
sl@0
     7
//
sl@0
     8
// Initial Contributors:
sl@0
     9
// Nokia Corporation - initial contribution.
sl@0
    10
//
sl@0
    11
// Contributors:
sl@0
    12
//
sl@0
    13
// Description:
sl@0
    14
//
sl@0
    15
sl@0
    16
#include "SqlUtil.h"
sl@0
    17
#include "SqlBufFlat.h"
sl@0
    18
#include "OstTraceDefinitions.h"
sl@0
    19
sl@0
    20
/**
sl@0
    21
Sets the flat buffer pointer to NULL
sl@0
    22
*/
sl@0
    23
RSqlBufFlat::RSqlBufFlat() :
sl@0
    24
	iBuf(NULL),
sl@0
    25
	iMaxSize(0),
sl@0
    26
	iBufPtr(NULL, 0)
sl@0
    27
	{
sl@0
    28
	}
sl@0
    29
sl@0
    30
/**
sl@0
    31
"Resource acquisiton" method.
sl@0
    32
Sets the elements count of a new or already existing flat buffer.
sl@0
    33
sl@0
    34
The occupied memory won't be freed (in case of buffer resizing).
sl@0
    35
The buffer content is not preserved (in case of buffer resizing).
sl@0
    36
sl@0
    37
All elements set to have:
sl@0
    38
- invalid type;
sl@0
    39
- zero length;
sl@0
    40
- zero data position;
sl@0
    41
sl@0
    42
@param aCount Desired flat buffer elements count
sl@0
    43
sl@0
    44
@return KErrNone, The operation has completed successfully;
sl@0
    45
		KErrNoMemory, Out of memory condition has occured.
sl@0
    46
*/
sl@0
    47
TInt RSqlBufFlat::SetCount(TInt aCount)
sl@0
    48
	{
sl@0
    49
	__ASSERT_DEBUG(aCount >= 0, __SQLPANIC(ESqlPanicBadArgument));
sl@0
    50
	TInt headerSize = sizeof(RSqlBufFlat::TCell) * aCount;
sl@0
    51
	TInt newSize = headerSize + sizeof(RSqlBufFlat::TBufFlat);
sl@0
    52
	if(DoReAlloc(newSize) != KErrNone)
sl@0
    53
		{
sl@0
    54
		return KErrNoMemory;	
sl@0
    55
		}
sl@0
    56
	TBufFlat& buf = *iBuf;
sl@0
    57
	buf.iCount = aCount;
sl@0
    58
	buf.iHeaderSize = headerSize;
sl@0
    59
	buf.iSize = newSize;
sl@0
    60
	buf.iReserved = 0;
sl@0
    61
	DoInit();
sl@0
    62
	SQLFLATBUF_INVARIANT();
sl@0
    63
	return KErrNone;	
sl@0
    64
	}
sl@0
    65
sl@0
    66
/**
sl@0
    67
Reallocates the amount of the occupied by the flat buffer memory.
sl@0
    68
The operation preserves the content of the flat buffer.
sl@0
    69
Note: if the new size is less or equal to the max size of the buffer,
sl@0
    70
	  no memory will be allocated.
sl@0
    71
sl@0
    72
@param aSize Desired flat buffer size in bytes
sl@0
    73
sl@0
    74
@return KErrNone, The operation has completed successfully;
sl@0
    75
		KErrNoMemory, Out of memory condition has occured.
sl@0
    76
*/
sl@0
    77
TInt RSqlBufFlat::ReAlloc(TInt aSize)
sl@0
    78
	{
sl@0
    79
	SQLFLATBUF_INVARIANT();
sl@0
    80
	TInt err = DoReAlloc(aSize);
sl@0
    81
	SQLFLATBUF_INVARIANT();
sl@0
    82
	return err;
sl@0
    83
	}
sl@0
    84
sl@0
    85
/**
sl@0
    86
Cleans up the flat buffer content and 
sl@0
    87
frees the occupied memory if the memory is above KBufLimit.
sl@0
    88
The count of elements is preserved.
sl@0
    89
sl@0
    90
If the buffer size is bigger than KBufLimit,
sl@0
    91
the buffer will be reallocated and the buffer content - not preserved.
sl@0
    92
sl@0
    93
If the buffer size is less or equal to KBufLimit,
sl@0
    94
no memory will be reallocated and the buffer preserves its content.
sl@0
    95
sl@0
    96
It is guaranteed that the reallocated buffer will have the same address in the heap 
sl@0
    97
as the original one.
sl@0
    98
*/	
sl@0
    99
void RSqlBufFlat::ResetAndMinimize()
sl@0
   100
	{
sl@0
   101
	SQLFLATBUF_INVARIANT();
sl@0
   102
	Reset();
sl@0
   103
#ifdef _DEBUG	
sl@0
   104
	const TInt KBufLimit = Max((TInt)RSqlBufFlat::EExpandSize, SysDataSize());
sl@0
   105
	const TBufFlat* oldBuf = iBuf;
sl@0
   106
#else
sl@0
   107
	const TInt KBufLimit = Max((8 * 1024), SysDataSize());
sl@0
   108
#endif
sl@0
   109
	if(iMaxSize > KBufLimit)
sl@0
   110
		{
sl@0
   111
		iMaxSize = KBufLimit - 1;  //to force the reallocation
sl@0
   112
		(void)DoReAlloc(KBufLimit);//User::ReAlloc() does not fail if the new requested size is less than the current block size
sl@0
   113
		}
sl@0
   114
	__ASSERT_DEBUG(oldBuf == iBuf, __SQLPANIC(ESqlPanicInternalError));
sl@0
   115
	SQLFLATBUF_INVARIANT();
sl@0
   116
	}
sl@0
   117
sl@0
   118
/**
sl@0
   119
Cleans up the flat buffer content but does not free the occupied memory.
sl@0
   120
The count of elements is preserved.
sl@0
   121
sl@0
   122
All elements set to have:
sl@0
   123
- invalid type;
sl@0
   124
- zero length;
sl@0
   125
- zero data position;
sl@0
   126
sl@0
   127
The element count is preserved.
sl@0
   128
*/
sl@0
   129
void RSqlBufFlat::Reset()
sl@0
   130
	{
sl@0
   131
	SQLFLATBUF_INVARIANT();
sl@0
   132
	iBuf->iSize = SysDataSize();
sl@0
   133
	DoInit();
sl@0
   134
	SQLFLATBUF_INVARIANT();
sl@0
   135
	}
sl@0
   136
sl@0
   137
/**
sl@0
   138
Closes the flat bufer and frees the allocated memory.
sl@0
   139
*/
sl@0
   140
void RSqlBufFlat::Close()
sl@0
   141
	{
sl@0
   142
	User::Free(iBuf);
sl@0
   143
	iBuf = NULL;	
sl@0
   144
	}
sl@0
   145
sl@0
   146
/**
sl@0
   147
Sets the content of a field.
sl@0
   148
sl@0
   149
@param aIndex Field index
sl@0
   150
@param aType Field type
sl@0
   151
@param aData Field data, may be NULL
sl@0
   152
@param aDataLength Field data length, may be 0
sl@0
   153
sl@0
   154
@return KErrNone, The operation has completed successfully;
sl@0
   155
		KErrNoMemory, Out of memory condition has occured.
sl@0
   156
*/
sl@0
   157
TInt RSqlBufFlat::SetField(TInt aIndex, TInt aType, const void* aData, TInt aDataLength)
sl@0
   158
	{
sl@0
   159
	SQLFLATBUF_INVARIANT();
sl@0
   160
	__ASSERT_DEBUG((TUint)aIndex < iBuf->iCount, __SQLPANIC(ESqlPanicBadArgument));
sl@0
   161
	__ASSERT_DEBUG((TUint)aType < RSqlBufFlat::EMaxType, __SQLPANIC(ESqlPanicBadArgument));
sl@0
   162
	__ASSERT_DEBUG((TUint)aDataLength < RSqlBufFlat::EMaxLength, __SQLPANIC(ESqlPanicBadArgument));
sl@0
   163
	if(aData)						//field value "present"
sl@0
   164
		{
sl@0
   165
		__ASSERT_DEBUG(aDataLength >= 0, __SQLPANIC(ESqlPanicBadArgument));
sl@0
   166
		if(aDataLength > 0)
sl@0
   167
			{
sl@0
   168
			if(Reserve(aDataLength) != KErrNone)
sl@0
   169
				{
sl@0
   170
				return KErrNoMemory;	
sl@0
   171
				}
sl@0
   172
			}
sl@0
   173
		DoSet(aIndex, aType, aData, aDataLength);
sl@0
   174
		}
sl@0
   175
	else if(aDataLength == 0)		//data is NULL, length is 0 - "null" field
sl@0
   176
		{
sl@0
   177
		DoSet(aIndex, aType, NULL, 0);
sl@0
   178
		}
sl@0
   179
	else							//field value "not present"
sl@0
   180
		{
sl@0
   181
		RSqlBufFlat::TCell& cell = *(Header() + aIndex);
sl@0
   182
		cell.iBits = (TUint)(((TUint)aType << RSqlBufFlat::EWidthLen) | (TUint)aDataLength);
sl@0
   183
		cell.iPos = 0;
sl@0
   184
		}
sl@0
   185
	SQLFLATBUF_INVARIANT();
sl@0
   186
	return KErrNone;
sl@0
   187
	}
sl@0
   188
	
sl@0
   189
/**
sl@0
   190
Initialzies the flat buffer header.
sl@0
   191
All field set:
sl@0
   192
- invalid type;
sl@0
   193
- zero length;
sl@0
   194
- "Not present";
sl@0
   195
*/	
sl@0
   196
void RSqlBufFlat::DoInit()
sl@0
   197
	{
sl@0
   198
	TBufFlat& buf = *iBuf;
sl@0
   199
	__ASSERT_DEBUG(buf.iCount >= 0, __SQLPANIC(ESqlPanicInternalError));
sl@0
   200
	__ASSERT_DEBUG(buf.iSize <= iMaxSize, __SQLPANIC(ESqlPanicInternalError));
sl@0
   201
	__ASSERT_DEBUG(buf.iHeaderSize == sizeof(RSqlBufFlat::TCell) * buf.iCount, __SQLPANIC(ESqlPanicInternalError));
sl@0
   202
	if(buf.iHeaderSize > 0)
sl@0
   203
		{
sl@0
   204
		Mem::FillZ(Header(), buf.iHeaderSize);
sl@0
   205
		}
sl@0
   206
	}
sl@0
   207
sl@0
   208
/**
sl@0
   209
Reallocates the amount of the occupied by the flat buffer memory
sl@0
   210
(only in case the requested size is bigger than the buffer size or the buffer does not exist).
sl@0
   211
The operation preserves the content of the flat buffer.
sl@0
   212
sl@0
   213
@param aSize Desired flat buffer size in bytes.
sl@0
   214
sl@0
   215
@return KErrNone, The operation has completed successfully;
sl@0
   216
		KErrNoMemory, Out of memory condition has occured.
sl@0
   217
*/
sl@0
   218
TInt RSqlBufFlat::DoReAlloc(TInt aSize)
sl@0
   219
	{
sl@0
   220
	if(!iBuf || iMaxSize < aSize)
sl@0
   221
		{
sl@0
   222
		//Calculate buffer new size (sometimes allocates more, for example, if
sl@0
   223
		//aSize % RSqlBufFlat::EExpandSize == 0, then one more RSqlBufFlat::EExpandSize page is allocated).
sl@0
   224
		TInt newSize = (aSize / RSqlBufFlat::EExpandSize + 1) * RSqlBufFlat::EExpandSize;
sl@0
   225
		RSqlBufFlat::TBufFlat* newBuf = static_cast <RSqlBufFlat::TBufFlat*> (User::ReAlloc(iBuf, newSize));
sl@0
   226
		if(!newBuf)
sl@0
   227
			{
sl@0
   228
			return KErrNoMemory;	
sl@0
   229
			}
sl@0
   230
		iBuf = newBuf;
sl@0
   231
		iMaxSize = newSize;
sl@0
   232
		}
sl@0
   233
	return KErrNone;
sl@0
   234
	}
sl@0
   235
sl@0
   236
/**
sl@0
   237
Initialzes a flat buffer field.
sl@0
   238
A memory for the field data has to be allocated before the call.
sl@0
   239
sl@0
   240
@param aIndex Field index
sl@0
   241
@param aType Field type
sl@0
   242
@param aData Field data, may be NULL
sl@0
   243
@param aDataLength Field data length, may be 0
sl@0
   244
*/
sl@0
   245
void RSqlBufFlat::DoSet(TInt aIndex, TInt aType, const void* aData, TInt aDataLength)
sl@0
   246
	{
sl@0
   247
	TBufFlat& buf = *iBuf;
sl@0
   248
	__ASSERT_DEBUG((TUint)aDataLength < RSqlBufFlat::EMaxLength, __SQLPANIC(ESqlPanicBadArgument));
sl@0
   249
	__ASSERT_DEBUG(aDataLength > 0 ? aData != NULL : ETrue, __SQLPANIC(ESqlPanicBadArgument));
sl@0
   250
	__ASSERT_DEBUG(aDataLength <= (iMaxSize - buf.iSize), __SQLPANIC(ESqlPanicInternalError));
sl@0
   251
	__ASSERT_DEBUG(::IsAligned8(buf.iSize), __SQLPANIC(ESqlPanicInternalError));
sl@0
   252
	RSqlBufFlat::TCell& cell = *(Header() + aIndex);
sl@0
   253
	cell.iBits = (TUint)(((TUint)aType << RSqlBufFlat::EWidthLen) | (TUint)aDataLength);
sl@0
   254
	cell.iPos = 1;	//not 0, because 0 means "not present"
sl@0
   255
	if(aDataLength > 0)	//for fields with length > 0 set the data and reinitalize cell.iPos
sl@0
   256
		{
sl@0
   257
#ifdef _DEBUG
sl@0
   258
		Mem::Copy(reinterpret_cast <TUint8*> (iBuf) + buf.iSize, &KSqlBufFlatMagicValue, sizeof(KSqlBufFlatMagicValue));
sl@0
   259
		buf.iSize += sizeof(KSqlBufFlatMagicValue);
sl@0
   260
#endif		
sl@0
   261
		cell.iPos = buf.iSize - sizeof(RSqlBufFlat::TBufFlat);
sl@0
   262
		Mem::Copy(reinterpret_cast <TUint8*> (iBuf) + buf.iSize, reinterpret_cast <const TUint8*> (aData), aDataLength);
sl@0
   263
		buf.iSize += ::AlignedLen8(aDataLength);	//align the next field start position
sl@0
   264
				                                  	//it is guaranteed that this "+" operation will not make iSize bigger than
sl@0
   265
				                                   	//iMaxSize, because the memory allocations are 8-byte aligned 
sl@0
   266
				                                   	//(even RSqlBufFlat::EExpandSize aligned)
sl@0
   267
		}
sl@0
   268
	}
sl@0
   269
sl@0
   270
/**
sl@0
   271
Makes sure that the flat buffer has enough free space for a block of data with "aLength" length.
sl@0
   272
The function may reallocated the buffer if there is not enough space.
sl@0
   273
sl@0
   274
@param aLength The requested free memory length.
sl@0
   275
sl@0
   276
@return KErrNone, The operation has completed successfully;
sl@0
   277
		KErrNoMemory, Out of memory condition has occured.
sl@0
   278
*/
sl@0
   279
TInt RSqlBufFlat::Reserve(TInt aLength)
sl@0
   280
	{
sl@0
   281
#ifdef _DEBUG
sl@0
   282
	TInt diff = aLength + sizeof(KSqlBufFlatMagicValue) - Available();
sl@0
   283
#else
sl@0
   284
	TInt diff = aLength - Available();
sl@0
   285
#endif		
sl@0
   286
	return diff > 0 ? DoReAlloc(iMaxSize + diff) : KErrNone;
sl@0
   287
	}
sl@0
   288
sl@0
   289
#ifdef _DEBUG
sl@0
   290
/**
sl@0
   291
Panics in _DEBUG mode if the flat buffer content is inconsistent.
sl@0
   292
*/
sl@0
   293
void RSqlBufFlat::Invariant() const
sl@0
   294
	{
sl@0
   295
	__ASSERT_DEBUG(iBuf != NULL, __SQLPANIC(ESqlPanicInternalError));
sl@0
   296
	const TBufFlat& buf = *iBuf;
sl@0
   297
	__ASSERT_DEBUG(buf.iCount >= 0, __SQLPANIC(ESqlPanicInternalError));
sl@0
   298
	__ASSERT_DEBUG(buf.iHeaderSize == sizeof(RSqlBufFlat::TCell) * buf.iCount, __SQLPANIC(ESqlPanicInternalError));
sl@0
   299
	__ASSERT_DEBUG(::IsAligned8(buf.iSize), __SQLPANIC(ESqlPanicInternalError));
sl@0
   300
	__ASSERT_DEBUG(buf.iSize >= buf.iHeaderSize + sizeof(RSqlBufFlat::TBufFlat), __SQLPANIC(ESqlPanicInternalError));
sl@0
   301
	__ASSERT_DEBUG(buf.iSize <= iMaxSize, __SQLPANIC(ESqlPanicInternalError));
sl@0
   302
	__ASSERT_DEBUG(buf.iSize <= User::AllocLen(iBuf), __SQLPANIC(ESqlPanicInternalError));	
sl@0
   303
	for(TInt i=0;i<(TInt)buf.iCount;++i)
sl@0
   304
		{
sl@0
   305
		const RSqlBufFlat::TCell& cell = *((reinterpret_cast <const RSqlBufFlat::TCell*> (iBuf + 1)) + i);
sl@0
   306
		__ASSERT_DEBUG(cell.Type() < RSqlBufFlat::EMaxType, __SQLPANIC(ESqlPanicInternalError));
sl@0
   307
		if(cell.Size() > 0 && cell.iPos >= buf.iHeaderSize) //only for present fields with length > 0
sl@0
   308
			{
sl@0
   309
			__ASSERT_DEBUG((TUint)cell.Size() <= buf.iSize, __SQLPANIC(ESqlPanicInternalError));
sl@0
   310
			__ASSERT_DEBUG(cell.iPos < (buf.iSize - sizeof(RSqlBufFlat::TBufFlat)), __SQLPANIC(ESqlPanicInternalError));
sl@0
   311
			TUint64 val = *(TUint64*)(reinterpret_cast <TUint8*> (iBuf) + cell.iPos + sizeof(RSqlBufFlat::TBufFlat) - sizeof(KSqlBufFlatMagicValue));
sl@0
   312
			__ASSERT_DEBUG(val == KSqlBufFlatMagicValue, __SQLPANIC(ESqlPanicInternalError));
sl@0
   313
			}
sl@0
   314
		}
sl@0
   315
	}
sl@0
   316
#endif//_DEBUG