Update contrib.
1 // Copyright (c) 2008-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.
18 #include "SqlAssert.h"
19 #include "SqlDatabaseImpl.h"
21 #include "OstTraceDefinitions.h"
22 #ifdef OST_TRACE_COMPILER_IN_USE
23 #include "SqlBlobTraces.h"
25 #include "SqlTraceDef.h"
27 // The maximum time (100 milliseconds) that a block of data is allowed to take to be read/written by TSqlBlob.
28 // If the time taken is longer than this value then the next block size used will be less
29 const TInt KTimeThresholdInMicroSecs = 100000;
30 // The largest block size used by TSqlBlob is 32Kb
31 const TInt KLargestBlockSize = 32 * 1024;
32 // The client-side buffer will be used instead if the calculated block size is 2Kb or less
33 const TInt KUseClientBufferThreshold = 2 * 1024;
35 // Prepare an IPC buffer containing the parameter values
36 static HBufC8* PrepareIpcParamBufLC(const TDesC& aTableName, const TDesC& aColumnName, TInt64 aRowId,
37 TBool aReadOnly, const TDesC& aDbName)
39 HBufC8* ipcPrmBuf = HBufC8::NewLC(aTableName.Size() + sizeof(TDesC) +
40 aColumnName.Size() + sizeof(TDesC) +
41 sizeof(aRowId) + sizeof(aReadOnly) +
42 aDbName.Size() + sizeof(TDesC) + sizeof(TInt32) * 3);
43 TPtr8 ipcPrmDes = ipcPrmBuf->Des();
44 RDesWriteStream strm(ipcPrmDes);
46 strm << static_cast <TInt32> (aTableName.Length())
48 << static_cast <TInt32> (aColumnName.Length())
51 << static_cast<TInt32>(aReadOnly)
52 << static_cast <TInt32> (aDbName.Length())
59 // Create an IPC stream, sending the IPC buffer containing the parameter values
60 static MStreamBuf* CreateIpcStreamL(RSqlDbSession& aDbSession, const TDesC8& aIpcPrmBuf)
63 ipcArgs.Set(0, aIpcPrmBuf.Length());
64 ipcArgs.Set(1, &aIpcPrmBuf);
65 HIpcBuf* ipcBuf = HIpcBuf::NewL(aDbSession, ::MakeMsgCode(ESqlSrvDbBlobSource, ESqlSrvNoHandle, 0), ipcArgs);
69 // Used by TSqlBlob to calculate the appropriate block size to use for the next transfer of blob data
70 static TInt CalculateBlockSize(TInt aRemainingDataSize, TInt aPreviousBlockSize, TBool aTimeThresholdBreached)
72 TInt blockSize = aPreviousBlockSize;
73 if(aTimeThresholdBreached)
75 // The time threshold was breached using the previous block size
76 // so we will reduce the block size by a factor of 2
78 if(blockSize <= KUseClientBufferThreshold)
80 // Just use the client-side IPC buffer size if things get to this stage
81 blockSize = KIpcBufSize;
84 return aRemainingDataSize <= blockSize ? aRemainingDataSize : blockSize;
87 // Retrieve the data in blocks
88 static void DoReadInBlocksL(RSqlBlobReadStream& aStrm, TDes8& aDestBuffer, TInt aBlobSize)
90 TInt remainingDataSize = aBlobSize;
91 TBool timeThresholdBreached = EFalse;
92 TInt blockSize = KLargestBlockSize;
93 TPtr8 ptr((TUint8*)aDestBuffer.Ptr(), aDestBuffer.MaxSize());
95 while(remainingDataSize > 0)
97 // Calculate the block size to use for this iteration, based on
98 // how much data is remaining and how fast the previous read was
99 blockSize = CalculateBlockSize(remainingDataSize, blockSize, timeThresholdBreached);
101 // Read the block of data, timing how long it takes
104 aStrm.ReadL(ptr, blockSize);
106 TTimeIntervalMicroSeconds readTime = t2.MicroSecondsFrom(t1);
107 timeThresholdBreached = (readTime.Int64() > KTimeThresholdInMicroSecs);
109 // Update how much data is still to be read, and the buffer pointer
110 remainingDataSize -= blockSize;
111 __ASSERT_DEBUG(remainingDataSize >= 0, __SQLPANIC2(ESqlPanicInternalError));
112 ptr.Set((TUint8*)(ptr.Ptr() + blockSize), 0, remainingDataSize);
114 aDestBuffer.SetLength(aBlobSize);
117 // Send the data in blocks
118 static void DoWriteInBlocksL(RSqlBlobWriteStream& aStrm, const TDesC8& aData, TInt aDataSize)
120 TInt remainingDataSize = aDataSize;
121 TBool timeThresholdBreached = EFalse;
122 TInt blockSize = KLargestBlockSize;
124 while(remainingDataSize > 0)
126 // Calculate the block size to use for this iteration, based on
127 // how much data is remaining and how fast the previous write was
128 blockSize = CalculateBlockSize(remainingDataSize, blockSize, timeThresholdBreached);
130 // Write the block of data, timing how long it takes
133 aStrm.WriteL(aData.Right(remainingDataSize), blockSize);
135 TTimeIntervalMicroSeconds writeTime = t2.MicroSecondsFrom(t1);
136 timeThresholdBreached = (writeTime.Int64() > KTimeThresholdInMicroSecs);
138 // Update how much data is still to be written
139 remainingDataSize -= blockSize;
140 __ASSERT_DEBUG(remainingDataSize >= 0, __SQLPANIC2(ESqlPanicInternalError));
145 // The data is returned in a UTF-8 buffer that is allocated on the heap
146 static HBufC8* ReadLC(RSqlDatabase& aDb,
147 const TDesC& aTableName,
148 const TDesC& aColumnName,
150 const TDesC& aDbName)
152 RSqlBlobReadStream strm;
153 CleanupClosePushL(strm);
154 strm.OpenL(aDb, aTableName, aColumnName, aRowId, aDbName);
155 TInt blobSize = strm.SizeL(); // size of the blob, in bytes
156 HBufC8* buf = HBufC8::NewLC(blobSize);
157 TPtr8 ptr = buf->Des();
158 DoReadInBlocksL(strm, ptr, blobSize);
159 CleanupStack::Pop(); // buf
160 CleanupStack::PopAndDestroy(); // strm
161 CleanupStack::PushL(buf);
165 // The data is returned in the UTF-8 buffer passed in
166 static void ReadL(RSqlDatabase& aDb,
167 const TDesC& aTableName,
168 const TDesC& aColumnName,
171 const TDesC& aDbName)
173 RSqlBlobReadStream strm;
174 CleanupClosePushL(strm);
175 strm.OpenL(aDb, aTableName, aColumnName, aRowId, aDbName);
176 TInt blobSize = strm.SizeL(); // size of the blob, in bytes
177 if(blobSize > aBuffer.MaxSize())
179 __SQLLEAVE2(KErrOverflow);
181 DoReadInBlocksL(strm, aBuffer, blobSize);
182 CleanupStack::PopAndDestroy(); // strm
185 // The data in the client-specified UTF-8 buffer is written
186 static void WriteL(RSqlDatabase& aDb,
187 const TDesC& aTableName,
188 const TDesC& aColumnName,
191 const TDesC& aDbName)
193 RSqlBlobWriteStream strm;
194 CleanupClosePushL(strm);
195 strm.OpenL(aDb, aTableName, aColumnName, aRowId, aDbName);
196 TInt dataSize = aData.Size(); // size of the data, in bytes
197 TInt blobSize = strm.SizeL(); // size of the blob, in bytes
198 if(dataSize > blobSize)
200 __SQLLEAVE2(KErrEof);
204 DoWriteInBlocksL(strm, aData, dataSize);
206 CleanupStack::PopAndDestroy(); // strm
209 ////////////////////////////////////////////////////////////////////////////////
212 Gives access to a blob as a read-only stream of bytes.
214 @param aDb A connection to the database that contains the blob
215 @param aTableName The name of the table that contains the blob
216 @param aColumnName The name of the column that contains the blob
217 @param aRowId The ROWID of the record that contains the blob,
218 or KSqlLastInsertedRowId if the last inserted ROWID
219 of the specified database connection is to be used
220 @param aDbName The name of the attached database if the blob is
221 contained in an attached database
223 @leave KSqlErrGeneral, Invalid database connection or table name or column name
224 or column type or ROWID or database name;
225 KErrNoMemory, An out of memory condition occurred;
226 KErrArgument, The ROWID is zero or negative or a UTF-16 to UTF-8 string conversion failed;
227 KErrBadName, The table name, column name or database name has an invalid length;
228 KErrPermissionDenied, The client does not have the required security capabilites for this operation;
229 Note that the function may also leave with some other system wide errors or
230 database specific errors categorised as ESqlDbError.
232 @panic SqlDb 2 The database object is not yet created
233 @panic SqlDb 3 Server failed to create a handle to the blob
234 @panic SqlDb 4 In _DEBUG mode. Bad parameter value
235 @panic SqlDb 7 In _DEBUG mode. NULL blob handle
237 @capability None, if the aDb parameter represents a handle which operates on a non-secure database;
238 RSqlSecurityPolicy::EReadPolicy or
239 RSqlSecurityPolicy::ESchemaPolicy database policy type, if the blob belongs to a secure database;
241 EXPORT_C void RSqlBlobReadStream::OpenL(RSqlDatabase& aDb, const TDesC& aTableName, const TDesC& aColumnName,
242 TInt64 aRowId, const TDesC& aDbName)
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));
245 HBufC8* ipcPrmBuf = ::PrepareIpcParamBufLC(aTableName, aColumnName, aRowId, ETrue, aDbName);
246 MStreamBuf* strm = ::CreateIpcStreamL(aDb.Impl().Session(), ipcPrmBuf->Des());
248 CleanupStack::PopAndDestroy(ipcPrmBuf);
249 SQL_TRACE_BORDER(OstTraceExt2(TRACE_BORDER, RSQLBLOBREADSTREAM_OPENL_EXIT, "Exit;0x%x;RSqlBlobReadStream::OpenL;strm=0x%X", (TUint)this, (TUint)strm));
253 Returns the size of the blob object, in bytes.
255 @return The size of the blob object, in bytes
257 @leave One of the system-wide error codes
259 @panic SqlDb 2 The stream buffer is NULL
263 EXPORT_C TInt RSqlBlobReadStream::SizeL()
265 SQL_TRACE_BORDER(OstTrace1(TRACE_BORDER, RSQLBLOBREADSTREAM_SIZEL_ENTRY, "Entry;0x%X;RSqlBlobReadStream::SizeL;", (TUint)this));
266 MStreamBuf* src = Source();
267 __ASSERT_ALWAYS(src != NULL, __SQLPANIC(ESqlPanicInvalidObj));
268 TInt size = src->SizeL();
269 SQL_TRACE_BORDER(OstTraceExt2(TRACE_BORDER, RSQLBLOBREADSTREAM_SIZEL_EXIT, "Exit;0x%X;RSqlBlobReadStream::SizeL;size=%d", (TUint)this, size));
274 Gives access to a blob as a writeable stream of bytes.
276 @param aDb A connection to the database that contains the blob
277 @param aTableName The name of the table that contains the blob
278 @param aColumnName The name of the column that contains the blob
279 @param aRowId The ROWID of the record that contains the blob,
280 or KSqlLastInsertedRowId if the last inserted ROWID
281 of the specified database connection is to be used
282 @param aDbName The name of the attached database if the blob is
283 contained in an attached database
285 @leave KSqlErrGeneral, Invalid database connection or table name or column name or column
286 type or ROWID or database name, or the specified column is indexed;
287 KErrNoMemory, An out of memory condition occurred;
288 KErrArgument, The ROWID is zero or negative or a UTF-16 to UTF-8 string conversion failed;
289 KErrBadName, The table name, column name or database name has an invalid length;
290 KErrPermissionDenied, The client does not have the required security capabilites for this operation;
291 Note that the function may also leave with some other system wide errors or
292 database specific errors categorised as ESqlDbError.
294 @panic SqlDb 2 The database object is not yet created
295 @panic SqlDb 3 Server failed to create a handle to the blob
296 @panic SqlDb 4 In _DEBUG mode. Bad parameter value
297 @panic SqlDb 7 In _DEBUG mode. NULL blob handle
299 @capability None, if the aDb parameter represents a handle which operates on a non-secure database;
300 RSqlSecurityPolicy::EWritePolicy or
301 RSqlSecurityPolicy::ESchemaPolicy database policy type, if the blob belongs to a secure database;
303 EXPORT_C void RSqlBlobWriteStream::OpenL(RSqlDatabase& aDb, const TDesC& aTableName, const TDesC& aColumnName,
304 TInt64 aRowId, const TDesC& aDbName)
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));
307 HBufC8* ipcPrmBuf = ::PrepareIpcParamBufLC(aTableName, aColumnName, aRowId, EFalse, aDbName);
308 MStreamBuf* strm = ::CreateIpcStreamL(aDb.Impl().Session(), ipcPrmBuf->Des());
310 CleanupStack::PopAndDestroy(ipcPrmBuf);
311 SQL_TRACE_BORDER(OstTraceExt2(TRACE_BORDER, RSQLBLOBWRITESTREAM_OPENL_EXIT, "Exit;0x%x;RSqlBlobWriteStream::OpenL;strm=0x%X", (TUint)this, (TUint)strm));
315 Returns the size of the blob object, in bytes.
317 @return The size of the blob object, in bytes
319 @leave One of the system-wide error codes
321 @panic SqlDb 2 The stream buffer is NULL
325 EXPORT_C TInt RSqlBlobWriteStream::SizeL()
327 SQL_TRACE_BORDER(OstTrace1(TRACE_BORDER, RSQLBLOBWRITESTREAM_SIZEL_ENTRY, "Entry;0x%X;RSqlBlobWriteStream::SizeL;", (TUint)this));
328 MStreamBuf* sink = Sink();
329 __ASSERT_ALWAYS(sink != NULL, __SQLPANIC(ESqlPanicInvalidObj));
330 TInt size = sink->SizeL();
331 SQL_TRACE_BORDER(OstTraceExt2(TRACE_BORDER, RSQLBLOBWRITESTREAM_SIZEL_EXIT, "Exit;0x%X;RSqlBlobWriteStream::SizeL;size=%d", (TUint)this, size));
336 Retrieves the entire content of a blob and returns it to the client
337 in a heap allocated buffer which has been placed on the cleanup stack.
338 It is the responsibility of the client to destroy the returned buffer.
340 @param aDb A connection to the database that contains the blob
341 @param aTableName The name of the table that contains the blob
342 @param aColumnName The name of the column that contains the blob
343 @param aRowId The ROWID of the record that contains the blob,
344 or KSqlLastInsertedRowId if the last inserted ROWID
345 of the specified database connection is to be used
346 @param aDbName The name of the attached database if the blob is
347 contained in an attached database
349 @leave KSqlErrGeneral, Invalid database connection or table name or column name
350 or column type or ROWID or database name;
351 KErrNoMemory, An out of memory condition occurred;
352 KErrArgument, The ROWID is zero or negative or a UTF-16 to UTF-8 string conversion failed;
353 KErrBadName, The table name, column name or database name has an invalid length;
354 KErrPermissionDenied, The client does not have the required security capabilites for this operation;
355 Note that the function may also leave with some other system wide errors or
356 database specific errors categorised as ESqlDbError.
358 @panic SqlDb 2 The database object is not yet created
359 @panic SqlDb 3 Server failed to create a handle to the blob
360 @panic SqlDb 4 In _DEBUG mode. Bad parameter value
361 @panic SqlDb 7 In _DEBUG mode. NULL blob handle or miscalculation of remaining data to be read
363 @capability None, if the aDb parameter represents a handle which operates on a non-secure database;
364 RSqlSecurityPolicy::EReadPolicy or
365 RSqlSecurityPolicy::ESchemaPolicy database policy type, if the blob belongs to a secure database;
367 EXPORT_C HBufC8* TSqlBlob::GetLC(RSqlDatabase& aDb,
368 const TDesC& aTableName,
369 const TDesC& aColumnName,
371 const TDesC& aDbName)
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));
374 HBufC8* res = ReadLC(aDb, aTableName, aColumnName, aRowId, aDbName);
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()));
380 Retrieves the entire content of a blob into a client specified buffer.
382 @param aDb A connection to the database that contains the blob
383 @param aTableName The name of the table that contains the blob
384 @param aColumnName The name of the column that contains the blob
385 @param aRowId The ROWID of the record that contains the blob,
386 or KSqlLastInsertedRowId if the last inserted ROWID
387 of the specified database connection is to be used
388 @param aDbName The name of the attached database if the blob is
389 contained in an attached database
391 @return KSqlErrGeneral, Invalid database connection or table name or column name
392 or column type or ROWID or database name;
393 KErrNoMemory, An out of memory condition occurred;
394 KErrArgument, The ROWID is zero or negative or a UTF-16 to UTF-8 string conversion failed;
395 KErrOverflow, The specified buffer is not big enough to hold the entire blob;
396 KErrBadName, The table name, column name or database name has an invalid length;
397 KErrPermissionDenied, The client does not have the required security capabilites for this operation;
398 Note that the function may also leave with some other system wide errors or
399 database specific errors categorised as ESqlDbError.
401 @panic SqlDb 2 The database object is not yet created
402 @panic SqlDb 3 Server failed to create a handle to the blob
403 @panic SqlDb 4 In _DEBUG mode. Bad parameter value
404 @panic SqlDb 7 In _DEBUG mode. NULL blob handle or miscalculation of remaining data to be read
406 @capability None, if the aDb parameter represents a handle which operates on a non-secure database;
407 RSqlSecurityPolicy::EReadPolicy or
408 RSqlSecurityPolicy::ESchemaPolicy database policy type, if the blob belongs to a secure database;
410 EXPORT_C TInt TSqlBlob::Get(RSqlDatabase& aDb,
411 const TDesC& aTableName,
412 const TDesC& aColumnName,
415 const TDesC& aDbName)
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));
418 TRAPD(err, ReadL(aDb, aTableName, aColumnName, aBuffer, aRowId, aDbName));
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()));
424 Writes the data in a client specified buffer to a blob.
426 @param aDb A connection to the database that contains the blob
427 @param aTableName The name of the table that contains the blob
428 @param aColumnName The name of the column that contains the blob
429 @param aRowId The ROWID of the record that contains the blob,
430 or KSqlLastInsertedRowId if the last inserted ROWID
431 of the specified database connection is to be used
432 @param aDbName The name of the attached database if the blob is
433 contained in an attached database
435 @leave KSqlErrGeneral, Invalid database connection or table name or column name or column
436 type or ROWID or database name, or the specified column is indexed;
437 KErrNoMemory, An out of memory condition occurred;
438 KErrArgument, The ROWID is zero or negative or a UTF-16 to UTF-8 string conversion failed;
439 KErrEof, The data in the specified buffer is larger in size than the blob object;
440 KErrBadName, The table name, column name or database name has an invalid length;
441 KErrPermissionDenied, The client does not have the required security capabilites for this operation;
442 Note that the function may also leave with some other system wide errors or
443 database specific errors categorised as ESqlDbError.
445 @panic SqlDb 2 The database object is not yet created
446 @panic SqlDb 3 Server failed to create a handle to the blob
447 @panic SqlDb 4 In _DEBUG mode. Bad parameter value
448 @panic SqlDb 7 In _DEBUG mode. NULL blob handle or miscalculation of remaining data to be written
450 @capability None, if the aDb parameter represents a handle which operates on a non-secure database;
451 RSqlSecurityPolicy::EWritePolicy or
452 RSqlSecurityPolicy::ESchemaPolicy database policy type, if the blob belongs to a secure database;
454 EXPORT_C void TSqlBlob::SetL(RSqlDatabase& aDb,
455 const TDesC& aTableName,
456 const TDesC& aColumnName,
459 const TDesC& aDbName)
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));
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()));
463 WriteL(aDb, aTableName, aColumnName, aData, aRowId, aDbName);
464 SQL_TRACE_BORDER(OstTrace0(TRACE_BORDER, TSQLBLOB_SET_EXIT, "Exit;0;TSqlBlob::Set"));