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".
8 // Initial Contributors:
9 // Nokia Corporation - initial contribution.
16 #include "SqlSrvFileData.h"
17 #include "SqlSrvUtil.h"
18 #include "SqlAssert.h"
19 #include "SqlSrvStrings.h"
20 #include "SqlSrvResourceProfiler.h"
22 _LIT(KPrivateFmtStr, "\\private\\%08X\\");
25 Creates SQL server private data path on the specified drive.
27 The idea for calling it is to make sure that the server's private data path exists before making any other
28 operation - attempting to create a database file there for example.
30 @param aFs File session instance
31 @param aDriveNumber Drive number on which the private path has to be created
35 static void CreatePrivateDataPathL(RFs& aFs, TDriveNumber aDriveNumber)
38 __SQLLEAVE_IF_ERROR2(aFs.Drive(driveInfo, aDriveNumber));
39 if(!(driveInfo.iDriveAtt & KDriveAttRom))
41 TInt err = aFs.CreatePrivatePath(aDriveNumber);
42 if(err != KErrNone && err != KErrAlreadyExists)
50 @return Zero if aDbFileName is a non-secure file name (contains the path), non-zero otherwise.
54 static TBool IsSecureFileNameFmt(const TDesC& aDbFileName)
56 TParsePtrC parse(aDbFileName);//this call may panic if aDbFileName cannot be parsed, but SetL() already parsed it
57 return !parse.PathPresent();
61 The function parses thr database file name argument and extracts the SID from it (if the name contains SID).
62 The SID is expected to be found at position 0 of the file name and must have 8 hex digits.
64 @param aDbFileName Database file name
66 @return Database security UID or KNullUid if the database name does not contain SID.
70 static TUid ExtractSID(const TDesC& aDbFileName)
72 TParsePtrC parse(aDbFileName);//this call may panic if aDbFileName cannot be parsed, but SetL() already parsed it
73 TPtrC dbName = parse.Name();
74 TInt pos1 = dbName.Locate(TChar('['));
75 TInt pos2 = dbName.Locate(TChar(']'));
76 if(pos1 == 0 && pos2 == 9) //position 0 for '[', 8 digits SID, position 9 for ']'
78 TLex lex(dbName.Mid(pos1 + 1, pos2 - pos1 - 1));
80 TInt err = lex.Val(*(TUint32*)&securityUid, EHex);
90 @return ETrue if the aDbFileName argument contains aPrivatePath as a first directory in the file path, EFalse otherwise.
94 static TBool IsPrivatePathInFileName(const TDesC& aDbFileName, const TDesC& aPrivatePath)
96 TInt pos = aDbFileName.FindF(aPrivatePath);
97 return (TUint)pos <= (TUint)KMaxDriveName;
101 The method parses aFileName argument and constructs the full database file name (including the path) there.
102 The full file name will be constructed in aFileName input/output argument.
104 @param aDbFileName Input/Output. Database file name will be constructed there.
105 @param aSysDrivePrivatePath SQL server private path on the system drive.
106 @param aDrive Output parameter. The drive number.
108 @leave KErrBadName Missing file name.
110 @panic SqlDb 7 In _DEBUG mode - no drive in the final file path.
114 static void DoFullFileNameL(TDes& aDbFileName, const TDesC& aSysDrivePrivatePath, TDriveNumber& aDrive)
117 __SQLLEAVE_IF_ERROR2(parse.Set(aDbFileName, &aSysDrivePrivatePath, NULL));
118 if(!parse.NamePresent())
120 __SQLLEAVE2(KErrBadName);
122 aDbFileName.Copy(parse.FullName());
123 TPtrC driveName = parse.Drive();
124 __ASSERT_DEBUG(driveName.Length() > 0, __SQLPANIC2(ESqlPanicInternalError));
125 TInt driveNumber = -1;
126 __SQLLEAVE_IF_ERROR2(RFs::CharToDrive(driveName[0], driveNumber));
127 aDrive = static_cast <TDriveNumber> (driveNumber);
131 Extracts file name properties, such as secure/non-secure file name, secure UID (SID).
133 @param aDbFileName Database file name
134 @param aServerPrivatePath SQL ser ver private path
135 @param aIsSecureFileNameFmt Output. Initialized with non-zero if aDbFileName format is "[drive:]name"
136 @param aSecureUid Output. Database secure UID. KNullUid for non-secure databases.
140 static void GetFileNamePropertiesL(const TDesC& aDbFileName, const TDesC& aServerPrivatePath,
141 TBool& aIsSecureFileNameFmt, TUid& aSecureUid)
143 //If SQL server private path is in the file name - leave
144 if(::IsPrivatePathInFileName(aDbFileName, aServerPrivatePath))
146 __SQLLEAVE2(KErrArgument);
148 //Extract database SID from the name
149 aIsSecureFileNameFmt = ::IsSecureFileNameFmt(aDbFileName);
150 aSecureUid = KNullUid;
151 if(aIsSecureFileNameFmt)
153 aSecureUid = ::ExtractSID(aDbFileName);
158 Extracts configuration parameters from client's config string.
159 For the rules how decision is made which parameter has to be used - from the config file or from the config string,
160 please check the TSqlSrvConfig class' comments.
161 If the client config string (aConfigStr argument) is NULL, then the config file parameters will be used (if there is a config file)
162 or the build-time partameters. Again, check the TSqlSrvConfig class' comments.
165 @see TSqlSrvConfigParams
167 @param aConfigStr Client configuration string, can be NULL
168 @param aConfigParams Output parameter, the place where config parameters will be stored
169 @param aConfig TSqlSrvConfig object used for the production of the config parameters
175 static void ExtractConfigParamsL(const TDesC8* aConfigStr, TSqlSrvConfigParams& aConfigParams, const TSqlSrvConfig& aConfig)
177 TPtrC8 ptr(aConfigStr ? *aConfigStr : KNullDesC8());
178 aConfig.GetConfigParamsL(ptr, aConfigParams);
182 1. Reads the database file name which is in "aFileNameArgNum" argument of aMessage and
183 initializes with it iFileName.
184 2. Parses the file name and initializes iIsSecureFileNameFmt and iSecureUid.
185 3. Creates the full file name in iFileName.
186 4. Creates the server private directory on the related drive.
188 @leave KErrBadName, the database file name length is invalid (not in [1..KMaxFileName] range);
189 KErrArgument, the database file name contains the server private path;
190 KErrArgument, the database file name format is secure but the name does not contain SID.
192 @panic SqlDb 4 In _DEBUG mode. Invalid aFileNameArgNum value.
193 @panic SqlDb 7 In _DEBUG mode. Invalid TSqlSrvFileData object. Not initialized system drive and path.
195 void TSqlSrvFileData::SetL(const RMessage2& aMessage, TInt aFileNameLen, TInt aFileNameArgNum,
196 #ifdef SQLSRV_STARTUP_TEST
197 const TDesC& aDbFileName,
199 const TDesC8* aConfigStr)
201 __ASSERT_DEBUG((TUint)aFileNameArgNum < KMaxMessageArguments, __SQLPANIC(ESqlPanicBadArgument));
202 __ASSERT_DEBUG(iSysDrivePrivatePath.DriveAndPath().Length() > 0, __SQLPANIC(ESqlPanicInternalError));
204 if(aFileNameLen < 1 || aFileNameLen > KMaxFileName)
206 __SQLLEAVE(KErrBadName);
208 #ifdef SQLSRV_STARTUP_TEST
209 //To prevent compiler warning
211 aFileNameArgNum = aFileNameArgNum;
213 iFileName.Copy(aDbFileName);
215 aMessage.ReadL(aFileNameArgNum, iFileName);
217 SQLPROFILER_REPORT_IPC(ESqlIpcRead, (aFileNameLen * sizeof(TText)));
218 TParse parsedFileName;
219 __SQLLEAVE_IF_ERROR(parsedFileName.Set(iFileName, 0, 0));//prophylactic check, leave if the file name cannot be parsed
220 ::GetFileNamePropertiesL(iFileName, iSysDrivePrivatePath.Path(), iIsSecureFileNameFmt, iSecureUid);
221 ::DoFullFileNameL(iFileName, iSysDrivePrivatePath.DriveAndPath(), iDrive);
222 iFileName.Append(TChar(0));
223 if(iIsSecureFileNameFmt)
225 if(iSecureUid == KNullUid)
227 __SQLLEAVE(KErrArgument);
229 ::CreatePrivateDataPathL(iFs, iDrive);
231 iReadOnly = ::IsReadOnlyFileL(iFs, FileName());
232 ::ExtractConfigParamsL(aConfigStr, iConfigParams, iConfig);
236 1. Initializes iFileName with the database file name.
237 2. Initializes iDrive.
238 3. Checks that iFileName really refers to a file belonging to application's private data cage.
240 Since the file to be created/opened is a file which belongs to the client application's private data cage
241 and the file has been created/opened already on the client side, iFileName is formatted to contain useful
242 information for the OS layer, such as file handle, file session handle, etc. The information is passed
243 to the OS layer in this strange way (formatted string treted as a file name), because the infomation goes
244 through the SQLITE library first.
245 The format of iFileName is:
247 Bytes:01 2 10 11 20 Last byte (before the terminating 0)
248 |<R/O flag><RMessage2 pointer><drive><app SID><file_name><file_ext>|
251 '|' is a symbol which cannot be placed in normal file names, so here it is used as an indication that the
252 string is not a file name (the string contains other information - handles message pointers, etc).
254 @leave KErrBadName, the database file name length is invalid (not in [1..KMaxFileName] range);
255 KErrPermissionDenied, the database file name is not in the application's private data cage.
257 void TSqlSrvFileData::SetFromHandleL(const RMessage2& aMessage, const TDesC& aDbFileName, TBool aCreated, TBool aReadOnly,
258 const TDesC8* aConfigStr)
260 TParse parsedFileName;
261 __SQLLEAVE_IF_ERROR(parsedFileName.Set(aDbFileName, 0, 0));//prophylactic check, leave if the file name cannot be parsed
263 iReadOnly = aReadOnly;
264 iIsSecureFileNameFmt = EFalse;
265 iSecureUid = KNullUid;
266 iFileName.Copy(aDbFileName);
267 TParsePtrC parse(iFileName);
268 if(!parse.DrivePresent() || !parse.PathPresent())
270 __SQLLEAVE(KErrBadName);
272 //Get the drive number
273 TPtrC driveName = parse.Drive();
274 TInt driveNumber = -1;
275 __SQLLEAVE_IF_ERROR(RFs::CharToDrive(driveName[0], driveNumber));
276 iDrive = static_cast <TDriveNumber> (driveNumber);
277 ::CreatePrivateDataPathL(iFs, iDrive);
278 //Create in "buf" variable calling application's private data path
279 TBuf<KMaxFileName + 1> buf;
280 buf.Format(KPrivateFmtStr(), aMessage.SecureId().iId);
281 //Check that the file name refers to a file which is in the application's private data cage
282 TInt pos = iFileName.FindF(buf);
283 if((TUint)pos > (TUint)KMaxDriveName)
285 __SQLLEAVE(KErrPermissionDenied);
287 //Form a new unique name for the database. It will be used when creating transaction rollback files, etc.
288 TPtrC nameAndExt = parse.NameAndExt();
289 buf.Format(KFileHandleFmt(), iReadOnly ? 1 : 0, &aMessage, &driveName, aMessage.SecureId().iId, &nameAndExt);
291 ::ExtractConfigParamsL(aConfigStr, iConfigParams, iConfig);