os/persistentdata/persistentstorage/sql/SRC/Client/SqlBlob.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) 2008-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 <s32mem.h>
sl@0
    17
#include "SqlDb.h"
sl@0
    18
#include "SqlAssert.h"
sl@0
    19
#include "SqlDatabaseImpl.h"
sl@0
    20
#include "IPCBuf.h"
sl@0
    21
#include "OstTraceDefinitions.h"
sl@0
    22
#ifdef OST_TRACE_COMPILER_IN_USE
sl@0
    23
#include "SqlBlobTraces.h"
sl@0
    24
#endif
sl@0
    25
#include "SqlTraceDef.h"
sl@0
    26
sl@0
    27
// The maximum time (100 milliseconds) that a block of data is allowed to take to be read/written by TSqlBlob.
sl@0
    28
// If the time taken is longer than this value then the next block size used will be less
sl@0
    29
const TInt KTimeThresholdInMicroSecs = 100000; 
sl@0
    30
// The largest block size used by TSqlBlob is 32Kb
sl@0
    31
const TInt KLargestBlockSize = 32 * 1024; 
sl@0
    32
// The client-side buffer will be used instead if the calculated block size is 2Kb or less 
sl@0
    33
const TInt KUseClientBufferThreshold = 2 * 1024; 
sl@0
    34
sl@0
    35
// Prepare an IPC buffer containing the parameter values
sl@0
    36
static HBufC8* PrepareIpcParamBufLC(const TDesC& aTableName, const TDesC& aColumnName, TInt64 aRowId, 
sl@0
    37
								  TBool aReadOnly, const TDesC& aDbName)
sl@0
    38
	{
sl@0
    39
	HBufC8* ipcPrmBuf = HBufC8::NewLC(aTableName.Size() + sizeof(TDesC) + 
sl@0
    40
									  aColumnName.Size() + sizeof(TDesC) +
sl@0
    41
									  sizeof(aRowId) + sizeof(aReadOnly) + 
sl@0
    42
									  aDbName.Size() + sizeof(TDesC) + sizeof(TInt32) * 3);
sl@0
    43
	TPtr8 ipcPrmDes = ipcPrmBuf->Des();
sl@0
    44
	RDesWriteStream strm(ipcPrmDes);
sl@0
    45
	
sl@0
    46
	strm << static_cast <TInt32> (aTableName.Length()) 
sl@0
    47
	     << aTableName 
sl@0
    48
	     << static_cast <TInt32> (aColumnName.Length()) 
sl@0
    49
	     << aColumnName 
sl@0
    50
	     << aRowId 
sl@0
    51
	     << static_cast<TInt32>(aReadOnly) 
sl@0
    52
	     << static_cast <TInt32> (aDbName.Length()) 
sl@0
    53
	     << aDbName;
sl@0
    54
	strm.CommitL();
sl@0
    55
	strm.Close();
sl@0
    56
	return ipcPrmBuf;
sl@0
    57
	}
sl@0
    58
sl@0
    59
// Create an IPC stream, sending the IPC buffer containing the parameter values
sl@0
    60
static MStreamBuf* CreateIpcStreamL(RSqlDbSession& aDbSession, const TDesC8& aIpcPrmBuf)
sl@0
    61
	{
sl@0
    62
	TIpcArgs ipcArgs;
sl@0
    63
	ipcArgs.Set(0, aIpcPrmBuf.Length());
sl@0
    64
	ipcArgs.Set(1, &aIpcPrmBuf);
sl@0
    65
	HIpcBuf* ipcBuf = HIpcBuf::NewL(aDbSession, ::MakeMsgCode(ESqlSrvDbBlobSource, ESqlSrvNoHandle, 0), ipcArgs);
sl@0
    66
	return ipcBuf;
sl@0
    67
	}	
sl@0
    68
	
sl@0
    69
// Used by TSqlBlob to calculate the appropriate block size to use for the next transfer of blob data
sl@0
    70
static TInt CalculateBlockSize(TInt aRemainingDataSize, TInt aPreviousBlockSize, TBool aTimeThresholdBreached)
sl@0
    71
	{
sl@0
    72
	TInt blockSize = aPreviousBlockSize;
sl@0
    73
	if(aTimeThresholdBreached)
sl@0
    74
		{
sl@0
    75
		// The time threshold was breached using the previous block size
sl@0
    76
		// so we will reduce the block size by a factor of 2
sl@0
    77
		blockSize /= 2;
sl@0
    78
		if(blockSize <= KUseClientBufferThreshold)
sl@0
    79
			{
sl@0
    80
			// Just use the client-side IPC buffer size if things get to this stage
sl@0
    81
			blockSize = KIpcBufSize; 
sl@0
    82
			}
sl@0
    83
		}	
sl@0
    84
	return aRemainingDataSize <= blockSize ? aRemainingDataSize : blockSize;
sl@0
    85
	}
sl@0
    86
		
sl@0
    87
// Retrieve the data in blocks
sl@0
    88
static void DoReadInBlocksL(RSqlBlobReadStream& aStrm, TDes8& aDestBuffer, TInt aBlobSize)
sl@0
    89
	{
sl@0
    90
	TInt remainingDataSize = aBlobSize;	
sl@0
    91
	TBool timeThresholdBreached = EFalse;
sl@0
    92
	TInt blockSize = KLargestBlockSize;
sl@0
    93
	TPtr8 ptr((TUint8*)aDestBuffer.Ptr(), aDestBuffer.MaxSize());
sl@0
    94
	
sl@0
    95
	while(remainingDataSize > 0)
sl@0
    96
		{	
sl@0
    97
		// Calculate the block size to use for this iteration, based on 
sl@0
    98
		// how much data is remaining and how fast the previous read was
sl@0
    99
		blockSize = CalculateBlockSize(remainingDataSize, blockSize, timeThresholdBreached);
sl@0
   100
		
sl@0
   101
		// Read the block of data, timing how long it takes
sl@0
   102
		TTime t1, t2;
sl@0
   103
		t1.HomeTime();
sl@0
   104
		aStrm.ReadL(ptr, blockSize);
sl@0
   105
		t2.HomeTime();
sl@0
   106
		TTimeIntervalMicroSeconds readTime = t2.MicroSecondsFrom(t1);
sl@0
   107
		timeThresholdBreached = (readTime.Int64() > KTimeThresholdInMicroSecs);
sl@0
   108
	
sl@0
   109
		// Update how much data is still to be read, and the buffer pointer
sl@0
   110
		remainingDataSize -= blockSize;
sl@0
   111
		__ASSERT_DEBUG(remainingDataSize >= 0, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   112
		ptr.Set((TUint8*)(ptr.Ptr() + blockSize), 0, remainingDataSize);
sl@0
   113
		}	
sl@0
   114
	aDestBuffer.SetLength(aBlobSize);
sl@0
   115
	}
sl@0
   116
	
sl@0
   117
// Send the data in blocks
sl@0
   118
static void DoWriteInBlocksL(RSqlBlobWriteStream& aStrm, const TDesC8& aData, TInt aDataSize)
sl@0
   119
	{
sl@0
   120
	TInt remainingDataSize = aDataSize;	
sl@0
   121
	TBool timeThresholdBreached = EFalse;
sl@0
   122
	TInt blockSize = KLargestBlockSize;
sl@0
   123
	
sl@0
   124
	while(remainingDataSize > 0)
sl@0
   125
		{	
sl@0
   126
		// Calculate the block size to use for this iteration, based on 
sl@0
   127
		// how much data is remaining and how fast the previous write was
sl@0
   128
		blockSize = CalculateBlockSize(remainingDataSize, blockSize, timeThresholdBreached);
sl@0
   129
sl@0
   130
		// Write the block of data, timing how long it takes
sl@0
   131
		TTime t1, t2;
sl@0
   132
		t1.HomeTime();
sl@0
   133
		aStrm.WriteL(aData.Right(remainingDataSize), blockSize);
sl@0
   134
		t2.HomeTime();
sl@0
   135
		TTimeIntervalMicroSeconds writeTime = t2.MicroSecondsFrom(t1);
sl@0
   136
		timeThresholdBreached = (writeTime.Int64() > KTimeThresholdInMicroSecs);
sl@0
   137
	
sl@0
   138
		// Update how much data is still to be written
sl@0
   139
		remainingDataSize -= blockSize;
sl@0
   140
		__ASSERT_DEBUG(remainingDataSize >= 0, __SQLPANIC2(ESqlPanicInternalError));
sl@0
   141
		}	
sl@0
   142
	aStrm.CommitL();	
sl@0
   143
	}
sl@0
   144
	
sl@0
   145
// The data is returned in a UTF-8 buffer that is allocated on the heap
sl@0
   146
static HBufC8* ReadLC(RSqlDatabase& aDb, 	
sl@0
   147
				   	  const TDesC& aTableName, 
sl@0
   148
				   	  const TDesC& aColumnName, 
sl@0
   149
				   	  TInt64 aRowId,
sl@0
   150
				   	  const TDesC& aDbName)	
sl@0
   151
	{
sl@0
   152
	RSqlBlobReadStream strm;
sl@0
   153
	CleanupClosePushL(strm);
sl@0
   154
	strm.OpenL(aDb, aTableName, aColumnName, aRowId, aDbName);		
sl@0
   155
	TInt blobSize = strm.SizeL(); // size of the blob, in bytes
sl@0
   156
	HBufC8* buf = HBufC8::NewLC(blobSize);
sl@0
   157
	TPtr8 ptr = buf->Des();	
sl@0
   158
	DoReadInBlocksL(strm, ptr, blobSize);
sl@0
   159
	CleanupStack::Pop(); // buf
sl@0
   160
	CleanupStack::PopAndDestroy(); // strm
sl@0
   161
	CleanupStack::PushL(buf);
sl@0
   162
	return buf;	
sl@0
   163
	}
sl@0
   164
sl@0
   165
// The data is returned in the UTF-8 buffer passed in
sl@0
   166
static void ReadL(RSqlDatabase& aDb, 	
sl@0
   167
				  const TDesC& aTableName, 
sl@0
   168
				  const TDesC& aColumnName, 	
sl@0
   169
				  TDes8& aBuffer,
sl@0
   170
				  TInt64 aRowId,
sl@0
   171
				  const TDesC& aDbName)
sl@0
   172
	{	
sl@0
   173
	RSqlBlobReadStream strm;
sl@0
   174
	CleanupClosePushL(strm);
sl@0
   175
	strm.OpenL(aDb, aTableName, aColumnName, aRowId, aDbName);		
sl@0
   176
	TInt blobSize = strm.SizeL(); // size of the blob, in bytes
sl@0
   177
	if(blobSize > aBuffer.MaxSize())
sl@0
   178
		{
sl@0
   179
		__SQLLEAVE2(KErrOverflow);
sl@0
   180
		}	
sl@0
   181
	DoReadInBlocksL(strm, aBuffer, blobSize);
sl@0
   182
	CleanupStack::PopAndDestroy(); // strm
sl@0
   183
	}
sl@0
   184
	
sl@0
   185
// The data in the client-specified UTF-8 buffer is written
sl@0
   186
static void WriteL(RSqlDatabase& aDb, 	
sl@0
   187
				   const TDesC& aTableName, 
sl@0
   188
				   const TDesC& aColumnName,
sl@0
   189
				   const TDesC8& aData,	
sl@0
   190
				   TInt64 aRowId,
sl@0
   191
				   const TDesC& aDbName)
sl@0
   192
	{
sl@0
   193
	RSqlBlobWriteStream strm;
sl@0
   194
	CleanupClosePushL(strm);
sl@0
   195
	strm.OpenL(aDb, aTableName, aColumnName, aRowId, aDbName);
sl@0
   196
	TInt dataSize = aData.Size(); // size of the data, in bytes
sl@0
   197
	TInt blobSize = strm.SizeL(); // size of the blob, in bytes
sl@0
   198
	if(dataSize > blobSize)
sl@0
   199
		{
sl@0
   200
		__SQLLEAVE2(KErrEof);
sl@0
   201
		}	
sl@0
   202
	if(dataSize > 0)
sl@0
   203
		{
sl@0
   204
		DoWriteInBlocksL(strm, aData, dataSize);
sl@0
   205
		}	
sl@0
   206
	CleanupStack::PopAndDestroy(); // strm
sl@0
   207
	}
sl@0
   208
sl@0
   209
////////////////////////////////////////////////////////////////////////////////	
sl@0
   210
			
sl@0
   211
/**
sl@0
   212
Gives access to a blob as a read-only stream of bytes.
sl@0
   213
sl@0
   214
@param aDb        	  A connection to the database that contains the blob
sl@0
   215
@param aTableName     The name of the table that contains the blob
sl@0
   216
@param aColumnName    The name of the column that contains the blob
sl@0
   217
@param aRowId         The ROWID of the record that contains the blob,
sl@0
   218
					  or KSqlLastInsertedRowId if the last inserted ROWID 
sl@0
   219
					  of the specified database connection is to be used
sl@0
   220
@param aDbName  	  The name of the attached database if the blob is 
sl@0
   221
					  contained in an attached database
sl@0
   222
sl@0
   223
@leave KSqlErrGeneral, 		 Invalid database connection or table name or column name 
sl@0
   224
							 or column type or ROWID or database name;							 
sl@0
   225
	   KErrNoMemory, 		 An out of memory condition occurred;	   
sl@0
   226
	   KErrArgument, 		 The ROWID is zero or negative or a UTF-16 to UTF-8 string conversion failed;
sl@0
   227
  	   KErrBadName,  		 The table name, column name or database name has an invalid length;
sl@0
   228
	   KErrPermissionDenied, The client does not have the required security capabilites for this operation; 						 
sl@0
   229
       						 Note that the function may also leave with some other system wide errors or 
sl@0
   230
                     		 database specific errors categorised as ESqlDbError.
sl@0
   231
	     
sl@0
   232
@panic SqlDb 2 The database object is not yet created
sl@0
   233
@panic SqlDb 3 Server failed to create a handle to the blob
sl@0
   234
@panic SqlDb 4 In _DEBUG mode. Bad parameter value
sl@0
   235
@panic SqlDb 7 In _DEBUG mode. NULL blob handle
sl@0
   236
sl@0
   237
@capability None, if the aDb parameter represents a handle which operates on a non-secure database;
sl@0
   238
            RSqlSecurityPolicy::EReadPolicy or 
sl@0
   239
            RSqlSecurityPolicy::ESchemaPolicy database policy type, if the blob belongs to a secure database; 
sl@0
   240
*/		
sl@0
   241
EXPORT_C void RSqlBlobReadStream::OpenL(RSqlDatabase& aDb, const TDesC& aTableName, const TDesC& aColumnName, 
sl@0
   242
										TInt64 aRowId, const TDesC& aDbName)
sl@0
   243
	{
sl@0
   244
	SQL_TRACE_BORDER(OstTraceExt5(TRACE_BORDER, RSQLBLOBREADSTREAM_OPENL_ENTRY, "Entry;0;RSqlBlobReadStream::OpenL;aDb=0x%X;aTableName=%S;aColumnName=%S;aDbName=%S;aRowId=%lld", (TUint)&aDb, __SQLPRNSTR(aTableName), __SQLPRNSTR(aColumnName), __SQLPRNSTR(aDbName), aRowId));
sl@0
   245
	HBufC8* ipcPrmBuf = ::PrepareIpcParamBufLC(aTableName, aColumnName, aRowId, ETrue, aDbName);
sl@0
   246
	MStreamBuf* strm = ::CreateIpcStreamL(aDb.Impl().Session(), ipcPrmBuf->Des());
sl@0
   247
	Attach(strm);
sl@0
   248
	CleanupStack::PopAndDestroy(ipcPrmBuf);	
sl@0
   249
    SQL_TRACE_BORDER(OstTraceExt2(TRACE_BORDER, RSQLBLOBREADSTREAM_OPENL_EXIT, "Exit;0x%x;RSqlBlobReadStream::OpenL;strm=0x%X", (TUint)this, (TUint)strm));
sl@0
   250
	}
sl@0
   251
	
sl@0
   252
/**
sl@0
   253
Returns the size of the blob object, in bytes.
sl@0
   254
sl@0
   255
@return The size of the blob object, in bytes
sl@0
   256
sl@0
   257
@leave One of the system-wide error codes
sl@0
   258
sl@0
   259
@panic SqlDb 2 The stream buffer is NULL
sl@0
   260
	
sl@0
   261
@capability None
sl@0
   262
*/
sl@0
   263
EXPORT_C TInt RSqlBlobReadStream::SizeL()
sl@0
   264
	{
sl@0
   265
    SQL_TRACE_BORDER(OstTrace1(TRACE_BORDER, RSQLBLOBREADSTREAM_SIZEL_ENTRY, "Entry;0x%X;RSqlBlobReadStream::SizeL;", (TUint)this));
sl@0
   266
	MStreamBuf* src = Source();
sl@0
   267
	__ASSERT_ALWAYS(src != NULL, __SQLPANIC(ESqlPanicInvalidObj));
sl@0
   268
	TInt size = src->SizeL();
sl@0
   269
    SQL_TRACE_BORDER(OstTraceExt2(TRACE_BORDER, RSQLBLOBREADSTREAM_SIZEL_EXIT, "Exit;0x%X;RSqlBlobReadStream::SizeL;size=%d", (TUint)this, size));
sl@0
   270
	return size;
sl@0
   271
	}
sl@0
   272
sl@0
   273
/**
sl@0
   274
Gives access to a blob as a writeable stream of bytes.
sl@0
   275
sl@0
   276
@param aDb        	  A connection to the database that contains the blob
sl@0
   277
@param aTableName     The name of the table that contains the blob
sl@0
   278
@param aColumnName    The name of the column that contains the blob
sl@0
   279
@param aRowId         The ROWID of the record that contains the blob,
sl@0
   280
					  or KSqlLastInsertedRowId if the last inserted ROWID 
sl@0
   281
					  of the specified database connection is to be used
sl@0
   282
@param aDbName  	  The name of the attached database if the blob is 
sl@0
   283
					  contained in an attached database
sl@0
   284
sl@0
   285
@leave KSqlErrGeneral, 		 Invalid database connection or table name or column name or column 
sl@0
   286
							 type or ROWID or database name, or the specified column is indexed;
sl@0
   287
	   KErrNoMemory, 		 An out of memory condition occurred;	   
sl@0
   288
	   KErrArgument, 		 The ROWID is zero or negative or a UTF-16 to UTF-8 string conversion failed;
sl@0
   289
  	   KErrBadName,  		 The table name, column name or database name has an invalid length;
sl@0
   290
	   KErrPermissionDenied, The client does not have the required security capabilites for this operation; 						 
sl@0
   291
       						 Note that the function may also leave with some other system wide errors or 
sl@0
   292
                     		 database specific errors categorised as ESqlDbError.
sl@0
   293
	     
sl@0
   294
@panic SqlDb 2 The database object is not yet created
sl@0
   295
@panic SqlDb 3 Server failed to create a handle to the blob
sl@0
   296
@panic SqlDb 4 In _DEBUG mode. Bad parameter value
sl@0
   297
@panic SqlDb 7 In _DEBUG mode. NULL blob handle
sl@0
   298
sl@0
   299
@capability None, if the aDb parameter represents a handle which operates on a non-secure database;
sl@0
   300
            RSqlSecurityPolicy::EWritePolicy or 
sl@0
   301
            RSqlSecurityPolicy::ESchemaPolicy database policy type, if the blob belongs to a secure database; 
sl@0
   302
*/							
sl@0
   303
EXPORT_C void RSqlBlobWriteStream::OpenL(RSqlDatabase& aDb, const TDesC& aTableName, const TDesC& aColumnName, 
sl@0
   304
										 TInt64 aRowId, const TDesC& aDbName)
sl@0
   305
	{
sl@0
   306
    SQL_TRACE_BORDER(OstTraceExt5(TRACE_BORDER, RSQLBLOBWRITESTREAM_OPENL_ENTRY, "Entry;0;RSqlBlobWriteStream::OpenL;aDb=0x%X;aTableName=%S;aColumnName=%S;aDbName=%S;aRowId=%lld", (TUint)&aDb, __SQLPRNSTR(aTableName), __SQLPRNSTR(aColumnName), __SQLPRNSTR(aDbName), aRowId));
sl@0
   307
	HBufC8* ipcPrmBuf = ::PrepareIpcParamBufLC(aTableName, aColumnName, aRowId, EFalse, aDbName);
sl@0
   308
	MStreamBuf* strm = ::CreateIpcStreamL(aDb.Impl().Session(), ipcPrmBuf->Des());
sl@0
   309
	Attach(strm);
sl@0
   310
	CleanupStack::PopAndDestroy(ipcPrmBuf);		
sl@0
   311
    SQL_TRACE_BORDER(OstTraceExt2(TRACE_BORDER, RSQLBLOBWRITESTREAM_OPENL_EXIT, "Exit;0x%x;RSqlBlobWriteStream::OpenL;strm=0x%X", (TUint)this, (TUint)strm));
sl@0
   312
	}
sl@0
   313
sl@0
   314
/**
sl@0
   315
Returns the size of the blob object, in bytes.
sl@0
   316
sl@0
   317
@return The size of the blob object, in bytes
sl@0
   318
sl@0
   319
@leave One of the system-wide error codes
sl@0
   320
sl@0
   321
@panic SqlDb 2 The stream buffer is NULL
sl@0
   322
	
sl@0
   323
@capability None
sl@0
   324
*/
sl@0
   325
EXPORT_C TInt RSqlBlobWriteStream::SizeL()
sl@0
   326
	{
sl@0
   327
    SQL_TRACE_BORDER(OstTrace1(TRACE_BORDER, RSQLBLOBWRITESTREAM_SIZEL_ENTRY, "Entry;0x%X;RSqlBlobWriteStream::SizeL;", (TUint)this));
sl@0
   328
	MStreamBuf* sink = Sink();
sl@0
   329
	__ASSERT_ALWAYS(sink != NULL, __SQLPANIC(ESqlPanicInvalidObj));
sl@0
   330
	TInt size =  sink->SizeL();
sl@0
   331
    SQL_TRACE_BORDER(OstTraceExt2(TRACE_BORDER, RSQLBLOBWRITESTREAM_SIZEL_EXIT, "Exit;0x%X;RSqlBlobWriteStream::SizeL;size=%d", (TUint)this, size));
sl@0
   332
    return size;
sl@0
   333
	}
sl@0
   334
sl@0
   335
/**
sl@0
   336
Retrieves the entire content of a blob and returns it to the client 
sl@0
   337
in a heap allocated buffer which has been placed on the cleanup stack. 
sl@0
   338
It is the responsibility of the client to destroy the returned buffer.
sl@0
   339
sl@0
   340
@param aDb        	  A connection to the database that contains the blob
sl@0
   341
@param aTableName     The name of the table that contains the blob
sl@0
   342
@param aColumnName    The name of the column that contains the blob
sl@0
   343
@param aRowId         The ROWID of the record that contains the blob,
sl@0
   344
					  or KSqlLastInsertedRowId if the last inserted ROWID 
sl@0
   345
					  of the specified database connection is to be used
sl@0
   346
@param aDbName  	  The name of the attached database if the blob is 
sl@0
   347
					  contained in an attached database
sl@0
   348
sl@0
   349
@leave KSqlErrGeneral, 		 Invalid database connection or table name or column name 
sl@0
   350
							 or column type or ROWID or database name;							 
sl@0
   351
	   KErrNoMemory, 		 An out of memory condition occurred;	   
sl@0
   352
	   KErrArgument, 		 The ROWID is zero or negative or a UTF-16 to UTF-8 string conversion failed;
sl@0
   353
  	   KErrBadName,  		 The table name, column name or database name has an invalid length;
sl@0
   354
	   KErrPermissionDenied, The client does not have the required security capabilites for this operation; 						 
sl@0
   355
       						 Note that the function may also leave with some other system wide errors or 
sl@0
   356
                     		 database specific errors categorised as ESqlDbError.
sl@0
   357
	     
sl@0
   358
@panic SqlDb 2 The database object is not yet created
sl@0
   359
@panic SqlDb 3 Server failed to create a handle to the blob
sl@0
   360
@panic SqlDb 4 In _DEBUG mode. Bad parameter value
sl@0
   361
@panic SqlDb 7 In _DEBUG mode. NULL blob handle or miscalculation of remaining data to be read
sl@0
   362
sl@0
   363
@capability None, if the aDb parameter represents a handle which operates on a non-secure database;
sl@0
   364
            RSqlSecurityPolicy::EReadPolicy or 
sl@0
   365
            RSqlSecurityPolicy::ESchemaPolicy database policy type, if the blob belongs to a secure database; 
sl@0
   366
*/
sl@0
   367
EXPORT_C HBufC8* TSqlBlob::GetLC(RSqlDatabase& aDb, 	
sl@0
   368
					     	     const TDesC& aTableName, 
sl@0
   369
					     		 const TDesC& aColumnName, 	
sl@0
   370
					     		 TInt64 aRowId,
sl@0
   371
					     		 const TDesC& aDbName)
sl@0
   372
	{
sl@0
   373
    SQL_TRACE_BORDER(OstTraceExt5(TRACE_BORDER, TSQLBLOB_GETLC_ENTRY, "Entry;0;TSqlBlob::GetLC;aDb=0x%X;aTableName=%S;aColumnName=%S;aDbName=%S;aRowId=%lld", (TUint)&aDb, __SQLPRNSTR(aTableName), __SQLPRNSTR(aColumnName), __SQLPRNSTR(aDbName), aRowId));
sl@0
   374
	HBufC8* res = ReadLC(aDb, aTableName, aColumnName, aRowId, aDbName);
sl@0
   375
    SQL_TRACE_BORDER(OstTraceExt2(TRACE_BORDER, TSQLBLOB_GETLC_EXIT, "Exit;0;TSqlBlob::GetLC;res=0x%X;res->Size()=%d", (TUint)res, res->Des().Size()));
sl@0
   376
	return res;
sl@0
   377
	}
sl@0
   378
	
sl@0
   379
/**
sl@0
   380
Retrieves the entire content of a blob into a client specified buffer.
sl@0
   381
sl@0
   382
@param aDb        	  A connection to the database that contains the blob
sl@0
   383
@param aTableName     The name of the table that contains the blob
sl@0
   384
@param aColumnName    The name of the column that contains the blob
sl@0
   385
@param aRowId         The ROWID of the record that contains the blob,
sl@0
   386
					  or KSqlLastInsertedRowId if the last inserted ROWID 
sl@0
   387
					  of the specified database connection is to be used
sl@0
   388
@param aDbName  	  The name of the attached database if the blob is 
sl@0
   389
					  contained in an attached database
sl@0
   390
sl@0
   391
@return KSqlErrGeneral, 	  Invalid database connection or table name or column name 
sl@0
   392
							  or column type or ROWID or database name;							 
sl@0
   393
	    KErrNoMemory, 		  An out of memory condition occurred;	   
sl@0
   394
	    KErrArgument, 		  The ROWID is zero or negative or a UTF-16 to UTF-8 string conversion failed;
sl@0
   395
 	    KErrOverflow, 		  The specified buffer is not big enough to hold the entire blob;
sl@0
   396
  	    KErrBadName,  		  The table name, column name or database name has an invalid length;
sl@0
   397
	    KErrPermissionDenied, The client does not have the required security capabilites for this operation; 						 
sl@0
   398
       						  Note that the function may also leave with some other system wide errors or 
sl@0
   399
                     		  database specific errors categorised as ESqlDbError.                    		 
sl@0
   400
	     
sl@0
   401
@panic SqlDb 2 The database object is not yet created
sl@0
   402
@panic SqlDb 3 Server failed to create a handle to the blob
sl@0
   403
@panic SqlDb 4 In _DEBUG mode. Bad parameter value
sl@0
   404
@panic SqlDb 7 In _DEBUG mode. NULL blob handle or miscalculation of remaining data to be read
sl@0
   405
sl@0
   406
@capability None, if the aDb parameter represents a handle which operates on a non-secure database;
sl@0
   407
            RSqlSecurityPolicy::EReadPolicy or 
sl@0
   408
            RSqlSecurityPolicy::ESchemaPolicy database policy type, if the blob belongs to a secure database; 
sl@0
   409
*/
sl@0
   410
EXPORT_C TInt TSqlBlob::Get(RSqlDatabase& aDb, 	
sl@0
   411
					 		const TDesC& aTableName, 
sl@0
   412
					 	    const TDesC& aColumnName, 	
sl@0
   413
					 	    TDes8& aBuffer,
sl@0
   414
					 		TInt64 aRowId,
sl@0
   415
					 	    const TDesC& aDbName)
sl@0
   416
	{
sl@0
   417
    SQL_TRACE_BORDER(OstTraceExt5(TRACE_BORDER, TSQLBLOB_GET_ENTRY, "Entry;0;TSqlBlob::Get;aDb=0x%X;aTableName=%S;aColumnName=%S;aDbName=%S;aRowId=%lld", (TUint)&aDb, __SQLPRNSTR(aTableName), __SQLPRNSTR(aColumnName), __SQLPRNSTR(aDbName), aRowId));
sl@0
   418
	TRAPD(err, ReadL(aDb, aTableName, aColumnName, aBuffer, aRowId, aDbName));
sl@0
   419
    SQL_TRACE_BORDER(OstTraceExt2(TRACE_BORDER, TSQLBLOB_GET_EXIT, "Exit;0;TSqlBlob::Get;aBuffer.Ptr()=0x%X;aBuffer.Size()=%d", (TUint)aBuffer.Ptr(), aBuffer.Size()));
sl@0
   420
	return err;	
sl@0
   421
	}
sl@0
   422
sl@0
   423
/**
sl@0
   424
Writes the data in a client specified buffer to a blob.
sl@0
   425
sl@0
   426
@param aDb        	  A connection to the database that contains the blob
sl@0
   427
@param aTableName     The name of the table that contains the blob
sl@0
   428
@param aColumnName    The name of the column that contains the blob
sl@0
   429
@param aRowId         The ROWID of the record that contains the blob,
sl@0
   430
					  or KSqlLastInsertedRowId if the last inserted ROWID 
sl@0
   431
					  of the specified database connection is to be used
sl@0
   432
@param aDbName  	  The name of the attached database if the blob is 
sl@0
   433
					  contained in an attached database
sl@0
   434
sl@0
   435
@leave KSqlErrGeneral, 		 Invalid database connection or table name or column name or column 
sl@0
   436
							 type or ROWID or database name, or the specified column is indexed;
sl@0
   437
	   KErrNoMemory, 		 An out of memory condition occurred;	   
sl@0
   438
	   KErrArgument, 		 The ROWID is zero or negative or a UTF-16 to UTF-8 string conversion failed;
sl@0
   439
	   KErrEof,	     		 The data in the specified buffer is larger in size than the blob object;
sl@0
   440
  	   KErrBadName,  		 The table name, column name or database name has an invalid length;
sl@0
   441
	   KErrPermissionDenied, The client does not have the required security capabilites for this operation; 						 
sl@0
   442
       						 Note that the function may also leave with some other system wide errors or 
sl@0
   443
                     		 database specific errors categorised as ESqlDbError.
sl@0
   444
	     
sl@0
   445
@panic SqlDb 2 The database object is not yet created
sl@0
   446
@panic SqlDb 3 Server failed to create a handle to the blob
sl@0
   447
@panic SqlDb 4 In _DEBUG mode. Bad parameter value
sl@0
   448
@panic SqlDb 7 In _DEBUG mode. NULL blob handle or miscalculation of remaining data to be written
sl@0
   449
sl@0
   450
@capability None, if the aDb parameter represents a handle which operates on a non-secure database;
sl@0
   451
            RSqlSecurityPolicy::EWritePolicy or 
sl@0
   452
            RSqlSecurityPolicy::ESchemaPolicy database policy type, if the blob belongs to a secure database; 
sl@0
   453
*/
sl@0
   454
EXPORT_C void TSqlBlob::SetL(RSqlDatabase& aDb, 	
sl@0
   455
					  		 const TDesC& aTableName, 
sl@0
   456
					  		 const TDesC& aColumnName,
sl@0
   457
					  		 const TDesC8& aData,	
sl@0
   458
					  		 TInt64 aRowId,
sl@0
   459
					  		 const TDesC& aDbName)
sl@0
   460
	{
sl@0
   461
    SQL_TRACE_BORDER(OstTraceExt5(TRACE_BORDER, TSQLBLOB_SET_ENTRY, "Entry;0;TSqlBlob::Set;aDb=0x%X;aTableName=%S;aColumnName=%S;aDbName=%S;aRowId=%lld", (TUint)&aDb, __SQLPRNSTR(aTableName), __SQLPRNSTR(aColumnName), __SQLPRNSTR(aDbName), aRowId));
sl@0
   462
    SQL_TRACE_BORDER(OstTraceExt2(TRACE_BORDER, TSQLBLOB_SET_ENTRYEXT, "EntryExt;0;TSqlBlob::Set;aData.Ptr=0x%X;aData.Size()=%d", (TUint)aData.Ptr(), aData.Size()));
sl@0
   463
	WriteL(aDb, aTableName, aColumnName, aData,	aRowId, aDbName);
sl@0
   464
    SQL_TRACE_BORDER(OstTrace0(TRACE_BORDER, TSQLBLOB_SET_EXIT, "Exit;0;TSqlBlob::Set"));
sl@0
   465
	}
sl@0
   466