os/persistentdata/persistentstorage/sql/SRC/Common/SqlUtil.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
     1 // Copyright (c) 2005-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 <e32svr.h>
    17 #include "SqlAssert.h"
    18 #include <sqldb.h>		//ESqlAtRow, ESqlAtEnd, ESqlErrGeneral
    19 #include "sqlite3.h"	//SQLITE_OK, SQLITE_ROW, SQLITE_DONE
    20 #include "OstTraceDefinitions.h"
    21 #ifdef OST_TRACE_COMPILER_IN_USE
    22 #include "SqlUtilTraces.h"
    23 #endif
    24 #include "SqlTraceDef.h"
    25 
    26 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
    27 
    28 const TInt KSqlLeavePanic = -359;//The (last-1) error code from the reserved area for the SQL component.
    29 
    30 #define UNUSED_ARG(arg) arg = arg
    31 #define UNUSED_DES(arg) arg
    32 
    33 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
    34 
    35 #if defined OST_TRACE_COMPILER_IN_USE &&  defined _SQL_RDEBUG_PRINT
    36 
    37 /**
    38 This class has been added here to avoid the crashes when _SQL_RDEBUG_PRINT macro is defined but the
    39 data to be printed out is too big and cannot fit into the buffer with size KSqlMaxPrnStrLen.
    40 @internalComponent   
    41 */
    42 class TSqlDes16Overflow : public TDes16Overflow
    43     {
    44 public:    
    45     virtual void Overflow(TDes16& /*aDes*/)
    46         {
    47         }
    48     };
    49 
    50 //Replaces:
    51 // 1) "%lld" with "%ld"
    52 // 2) "%s" with "%S"
    53 //These are the differences in format specification between RDebig::Print and OST functions.
    54 //The new format spec length should be less or equal than the old format spec length.
    55 static void ReplaceFmtSpec(TDes& aFormat, const TDesC& aFmtSpec, const TDesC& aNewFmtSpec)
    56 	{
    57 	TInt fmtLength = aFormat.Length();
    58 	const TInt KDiff = aFmtSpec.Length() - aNewFmtSpec.Length();
    59     TPtr ptr((TText*)aFormat.Ptr(), fmtLength, fmtLength);
    60     TInt pos;
    61     while((pos = ptr.Find(aFmtSpec)) >= 0)
    62     	{
    63 		ptr.Replace(pos, aFmtSpec.Length(), aNewFmtSpec);
    64 		fmtLength -= KDiff;
    65 		ptr.Set(ptr.MidTPtr(pos));
    66     	}
    67     aFormat.SetLength(fmtLength);
    68 	}
    69 
    70 void SqlPrintf(TInt /*aGroupName*/, TInt /*aTraceName*/, const char* aFormat, ...)
    71     {
    72     VA_LIST list;
    73     VA_START(list, aFormat);
    74     TBuf<128> format;
    75     _LIT(KTraceIdent, "SQL;");
    76     format.Copy(TPtrC8((const TUint8*)aFormat));
    77     format.Insert(0, KTraceIdent);
    78     format.Append(_L("\r\n"));
    79     _LIT(KOstI64Fmt, "%lld");
    80     _LIT(KDbgPrnI64Fmt, "%ld");
    81     ReplaceFmtSpec(format, KOstI64Fmt, KDbgPrnI64Fmt);
    82     _LIT(KOstDes8Fmt, "%s");
    83     _LIT(KDbgPrnDesFmt, "%S");
    84     ReplaceFmtSpec(format, KOstDes8Fmt, KDbgPrnDesFmt);
    85     TBuf<KSqlMaxPrnStrLen> buf;
    86     TSqlDes16Overflow overflowHandler;
    87     buf.AppendFormatList(format, list, &overflowHandler);
    88 #ifdef _SQL_RDEBUG_PRINT    
    89     RDebug::RawPrint(buf);
    90 #endif
    91     }
    92 
    93 const TDesC* SqlDes8to16Ptr(const TDesC8& aDes, TDes& aOut)
    94 	{
    95 	TPtrC8 ptr(aDes.Ptr(), Min(aDes.Length(), aOut.MaxLength()));
    96 	aOut.Copy(ptr);
    97 	return &aOut;
    98 	}
    99 
   100 #endif//defined OST_TRACE_COMPILER_IN_USE &&  defined _SQL_RDEBUG_PRINT 
   101 
   102 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   103 
   104 /**
   105 SQL panic category.
   106 
   107 @internalComponent
   108 */
   109 _LIT(KPanicCategory, "SqlDb");
   110 
   111 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   112 
   113 /**
   114 Panics the caller with aPanicCode panic code.
   115 The call will terminate the thread where it is called from.
   116 
   117 @param aPanicCode Panic code.
   118 
   119 @internalComponent
   120 */
   121 static void SqlPanic(TSqlPanic aPanicCode)
   122 	{
   123 	User::Panic(KPanicCategory, aPanicCode);
   124 	}
   125 	
   126 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   127 
   128 /**
   129 Panics the client with aPanicCode panic code.
   130 This function is used by the SQL server to panic the caller (the client).
   131 
   132 @param aMessage Client's message
   133 @param aPanicCode Panic code.
   134 
   135 @leave KSqlLeavePanic
   136 
   137 @return KErrNone
   138 
   139 @internalComponent
   140 */
   141 static TInt SqlPanicClientL(const RMessage2& aMessage, TSqlPanic aPanicCode)
   142 	{
   143 	aMessage.Panic(KPanicCategory, aPanicCode);
   144 	__SQLLEAVE2(KSqlLeavePanic);
   145 	return KErrNone;
   146 	}	
   147 
   148 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   149 
   150 /**
   151 The function prints out a "SQL panic" message to the console and panics the thread where it is called from.
   152 It gives a useful information about the found error together with the source file name and line number where
   153 it occurred.
   154 
   155 Note: this function  will output information regarding the panic only if _SQL_PANIC_TRACE_ENABLED macro is defined  
   156 
   157 @param aFile Source file name
   158 @param aLine Source line number
   159 @param aPanicCode Panic code
   160 @param aHandle Numeric value, uniquely identfying the leaving location (the "this" pointer for example)
   161 
   162 @return KErrNone
   163 
   164 @internalComponent
   165 */  
   166 TInt TSqlUtil::Panic(const TText* aFile, TInt aLine, TInt aPanicCode, TUint aHandle)
   167     {
   168 #if defined OST_TRACE_COMPILER_IN_USE && defined _SQL_PANIC_TRACE_ENABLED
   169     TPtrC fname(FileName(aFile));
   170     OstTraceExt5(TRACE_FATAL, TSQLUTIL_PANIC, "Panic;0x%X;%S;%d;%S;%d", aHandle, __SQLPRNSTR(fname), aLine, __SQLPRNSTR(KPanicCategory), aPanicCode);
   171 #else
   172     UNUSED_ARG(aFile);
   173     UNUSED_ARG(aLine);
   174     UNUSED_ARG(aHandle);
   175 #endif      
   176     ::SqlPanic(static_cast <TSqlPanic> (aPanicCode));
   177     return KErrNone;
   178     }
   179 
   180 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   181 
   182 /**
   183 The function prints out a "SQL leave" message to the console and leaves with aError error code.
   184 It gives a usefull information about the found error together with the source file name and line number where
   185 it occured.
   186 
   187 Note: this function  will output information regarding the panic only if _SQL_LEAVE_TRACE_ENABLED macro is defined  
   188 
   189 @param aFile Source file name
   190 @param aLine Source line number
   191 @param aError Error code
   192 @param aHandle Numeric value, uniquely identfying the leaving location (the "this" pointer for example)
   193 
   194 @internalComponent
   195 */  
   196 void TSqlUtil::Leave(const TText* aFile, TInt aLine, TInt aError, TUint aHandle)
   197     {
   198 #if defined OST_TRACE_COMPILER_IN_USE && defined _SQL_LEAVE_TRACE_ENABLED     
   199     TPtrC fname(FileName(aFile));
   200     OstTraceExt4(TRACE_ERROR, TSQLUTIL_LEAVE, "Leave;0x%X;%S;%d;Error=%d", aHandle, __SQLPRNSTR(fname), aLine, aError);
   201 #else
   202     UNUSED_ARG(aFile);
   203     UNUSED_ARG(aLine);
   204     UNUSED_ARG(aHandle);
   205 #endif
   206     User::Leave(aError);
   207     }
   208 
   209 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   210 
   211 /**
   212 The function prints out a "SQL leave" message to the console and leaves with aError error code, if it is 
   213 negative.
   214 It gives a usefull information about the found error together with the source file name and line number where
   215 it occured.
   216 
   217 Note: this function  will output information regarding the panic only if _SQL_LEAVE_TRACE_ENABLED macro is defined  
   218 
   219 @param aFile Source file name
   220 @param aLine Source line number
   221 @param aError Error code
   222 
   223 @internalComponent
   224 */  
   225 TInt TSqlUtil::LeaveIfError(const TText* aFile, TInt aLine, TInt aError, TUint aHandle)
   226     {
   227     if(aError < 0)
   228         {
   229         TSqlUtil::Leave(aFile, aLine, aError, aHandle);
   230         }
   231     return aError;
   232     }
   233 
   234 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   235 
   236 /**
   237 The function prints out a "SQL leave" message to the console and leaves with KErrNoMemory if 
   238 aPtr parameter is NULL.
   239 
   240 Note: this function  will output information regarding the panic only if _SQL_LEAVE_TRACE_ENABLED macro is defined  
   241 
   242 @param aFile Source file name
   243 @param aLine Source line number
   244 @param aPtr The pointer to be tested against NULL value.
   245 
   246 @internalComponent
   247 */  
   248 void* TSqlUtil::LeaveIfNull(const TText* aFile, TInt aLine, void* aPtr, TUint aHandle)
   249     {
   250     if(!aPtr)
   251         {
   252         TSqlUtil::Leave(aFile, aLine, KErrNoMemory, aHandle);
   253         }
   254     return aPtr;
   255     }
   256 
   257 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   258 
   259 /**
   260 The function is used by the SQL server.
   261 It prints out a "SQL panic" message to the console and panic the client.
   262 It gives a usefull information about the found error together with the source file name and line number where
   263 it occured.
   264 
   265 Note: this function  will output information regarding the panic only if _SQL_PANIC_TRACE_ENABLED macro is defined  
   266 
   267 @param aFile Source file name
   268 @param aLine Source line number
   269 @param aMessage The client message, which processing caused the panic.
   270 @param aPanicCode Error code
   271 
   272 @leave KSqlLeavePanic
   273 
   274 @return KErrNone;
   275 
   276 @internalComponent
   277 */  
   278 TInt TSqlUtil::PanicClientL(const TText* aFile, TInt aLine, const RMessage2& aMessage, TInt aPanicCode, TUint aHandle)
   279     {
   280 #if defined OST_TRACE_COMPILER_IN_USE && defined  _SQL_PANIC_TRACE_ENABLED
   281     TPtrC fname(FileName(aFile));
   282     OstTraceExt5(TRACE_FATAL, TSQLUTIL_PANICCLIENTL, "Panic;%X;%S;%d;%S;%d", aHandle, __SQLPRNSTR(fname), aLine, __SQLPRNSTR(KPanicCategory), aPanicCode);
   283 #else
   284     UNUSED_ARG(aFile);
   285     UNUSED_ARG(aLine);
   286     UNUSED_ARG(aHandle);
   287 #endif      
   288     return ::SqlPanicClientL(aMessage, static_cast <TSqlPanic> (aPanicCode));
   289     }
   290 
   291 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   292 
   293 /**
   294 Processes SQL database error code and OS error code and returns unified error code.
   295 If aSqlError == SQLITE_ROW then the function returns KSqlAtRow.
   296 If aSqlError == SQLITE_DONE then the function returns KSqlAtEnd.
   297 If aSqlError == SQLITE_NOMEM then the function returns KErrNoMemory.
   298 If aOsError != KErrNone then the function returns aOsError.
   299 Otherwise the function converts aSqlError to one of error codes in [KSqlErrGeneral..KSqlErrStmtExpired] range.
   300 
   301 @param aSqlError SQL database error code.
   302 @param aOsError OS error code.
   303 
   304 @return Database specific error code.
   305 
   306 @panic SqlDb 4 in debug mode - if aSqlError < 0
   307 @panic SqlDb 4 in debug mode - if aOsError > 0
   308 
   309 @internalComponent
   310 */
   311 TInt Sql2OsErrCode(TInt aSqlError, TInt aOsError)
   312 	{
   313 
   314 	__ASSERT_DEBUG(aSqlError >= SQLITE_OK && aOsError <= KErrNone, __SQLPANIC2(ESqlPanicBadArgument));
   315 	TInt err = KErrNone;
   316 	if(aOsError == KErrDiskFull)
   317 		{//Whatever is the aSqlError value, even SQLITE_OK, never ignore KErrDiskFull errors
   318 		 //(For example: ROLLBACK statement execution, when the disk is full).
   319 		err = aOsError;
   320 		}
   321 	else if(aSqlError == SQLITE_ROW)
   322 		{
   323 		err = KSqlAtRow;
   324 		}
   325 	else if(aSqlError == SQLITE_DONE)
   326 		{
   327 		err = KSqlAtEnd;
   328 		}
   329 	else if(aSqlError == SQLITE_NOMEM)
   330 		{
   331 		err = KErrNoMemory;
   332 		}
   333 	else if(aSqlError == SQLITE_AUTH)
   334 		{
   335 		err = KErrPermissionDenied;
   336 		}
   337 	else if(aSqlError == SQLITE_NOTADB)
   338 		{
   339 		err = KSqlErrNotDb;	
   340 		}
   341 	else if(aSqlError > SQLITE_OK)
   342 		{
   343 		err = aOsError != KErrNone ? aOsError : KSqlErrGeneral - aSqlError + 1;
   344 		}
   345 	return err;
   346 	}
   347 
   348 ///////////////////////////////////////////////////////////////////////////////////////////////////////////////////
   349 
   350 #if defined OST_TRACE_COMPILER_IN_USE && (defined _SQL_PANIC_TRACE_ENABLED || defined _SQL_LEAVE_TRACE_ENABLED) 
   351 
   352 /**
   353 The function creates and returns TPtrC object which points to aFile parameter.
   354 
   355 @param aFile File name
   356 @return TPtrC object pointing to aFile parameter.
   357 
   358 @internalComponent
   359 */	
   360 TPtrC TSqlUtil::FileName(const TText* aFile)
   361 	{
   362 	TPtrC p(aFile);
   363 	TInt ix = p.LocateReverse('\\');
   364 	if(ix<0)
   365 		ix=p.LocateReverse('/');
   366 	if(ix>=0)
   367 		p.Set(p.Mid(1+ix));
   368 	return p;
   369 	}
   370 
   371 #endif //defined OST_TRACE_COMPILER_IN_USE && (defined _SQL_PANIC_TRACE_ENABLED || defined _SQL_LEAVE_TRACE_ENABLED)