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 |
// SqlSrvStatementCollation.cpp
|
sl@0
|
15 |
//
|
sl@0
|
16 |
//
|
sl@0
|
17 |
|
sl@0
|
18 |
#include "SqlSrvCollation.h"
|
sl@0
|
19 |
#include "SqlAssert.h"
|
sl@0
|
20 |
#include "SqlUtil.h"
|
sl@0
|
21 |
#include "sqlite3.h"
|
sl@0
|
22 |
#include "SqliteSymbian.h" //sqlite3SymbianLastOsError()
|
sl@0
|
23 |
|
sl@0
|
24 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
25 |
///////////////////////////// Local const data ///////////////////////////////////////////////////////
|
sl@0
|
26 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
27 |
|
sl@0
|
28 |
//4 collated comparisons + 1 folded comparison
|
sl@0
|
29 |
const TInt KCollationMethodCount = 4 + 1;
|
sl@0
|
30 |
|
sl@0
|
31 |
//Names of the user defined collations (zero-terminated 16-bit strings)
|
sl@0
|
32 |
const TUint16* KCollationMethodName[KCollationMethodCount] =
|
sl@0
|
33 |
{
|
sl@0
|
34 |
(const TUint16*)L"CompareC0",
|
sl@0
|
35 |
(const TUint16*)L"CompareC1",
|
sl@0
|
36 |
(const TUint16*)L"CompareC2",
|
sl@0
|
37 |
(const TUint16*)L"CompareC3",
|
sl@0
|
38 |
(const TUint16*)L"CompareF"
|
sl@0
|
39 |
};
|
sl@0
|
40 |
|
sl@0
|
41 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
42 |
///////////////////////////// Local functions /////////////////////////////////////////////////////
|
sl@0
|
43 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
44 |
|
sl@0
|
45 |
//User-defined collation function prototype, accepted by SQLITE
|
sl@0
|
46 |
typedef TInt (*TCollationMethodPtr)(void* aUserDefinedData, TInt aByteSize1, const void* aPtr1, TInt aByteSize2, const void* aPtr2);
|
sl@0
|
47 |
|
sl@0
|
48 |
/**
|
sl@0
|
49 |
This function is used for collated/folded string comparisons,
|
sl@0
|
50 |
{aByteSize1, aPtr1} - string 1, UTF16 encoded. aByteSize1 - length in bytes.
|
sl@0
|
51 |
{aByteSize2, aPtr2} - string 2, UTF16 encoded. aByteSize2 - length in bytes.
|
sl@0
|
52 |
aLevel - 0,1,2,3 - collation level; -1 - folded string comparison;
|
sl@0
|
53 |
The function returns negative, zero or positive if the first string is less than, equal to, or
|
sl@0
|
54 |
greater than the second string.
|
sl@0
|
55 |
|
sl@0
|
56 |
@internalComponent
|
sl@0
|
57 |
*/
|
sl@0
|
58 |
static TInt Compare(TInt aLevel, TInt aByteSize1, const void* aPtr1, TInt aByteSize2, const void* aPtr2)
|
sl@0
|
59 |
{
|
sl@0
|
60 |
TPtrC16 ptr1(static_cast <const TUint16*> (aPtr1), (TUint)aByteSize1 /sizeof(TUint16));
|
sl@0
|
61 |
TPtrC16 ptr2(static_cast <const TUint16*> (aPtr2), (TUint)aByteSize2 /sizeof(TUint16));
|
sl@0
|
62 |
|
sl@0
|
63 |
return aLevel >= 0 ? ptr1.CompareC(ptr2, aLevel, NULL) : ptr1.CompareF(ptr2);
|
sl@0
|
64 |
}
|
sl@0
|
65 |
|
sl@0
|
66 |
/**
|
sl@0
|
67 |
User defined function which will be used by SQLITE engine for collated string comparisons,
|
sl@0
|
68 |
collation level 0.
|
sl@0
|
69 |
{aByteSize1, aPtr1} - string 1, UTF16 encoded. aByteSize1 - length in bytes.
|
sl@0
|
70 |
{aByteSize2, aPtr2} - string 2, UTF16 encoded. aByteSize2 - length in bytes.
|
sl@0
|
71 |
The function returns negative, zero or positive if the first string is less than, equal to, or
|
sl@0
|
72 |
greater than the second string.
|
sl@0
|
73 |
|
sl@0
|
74 |
@internalComponent
|
sl@0
|
75 |
*/
|
sl@0
|
76 |
static TInt CompareC0(void*, TInt aByteSize1, const void* aPtr1, TInt aByteSize2, const void* aPtr2)
|
sl@0
|
77 |
{
|
sl@0
|
78 |
return Compare(0, aByteSize1, aPtr1, aByteSize2, aPtr2);
|
sl@0
|
79 |
}
|
sl@0
|
80 |
|
sl@0
|
81 |
/**
|
sl@0
|
82 |
User defined function which will be used by SQLITE engine for collated string comparisons,
|
sl@0
|
83 |
collation level 1.
|
sl@0
|
84 |
{aByteSize1, aPtr1} - string 1, UTF16 encoded. aByteSize1 - length in bytes.
|
sl@0
|
85 |
{aByteSize2, aPtr2} - string 2, UTF16 encoded. aByteSize2 - length in bytes.
|
sl@0
|
86 |
The function returns negative, zero or positive if the first string is less than, equal to, or
|
sl@0
|
87 |
greater than the second string.
|
sl@0
|
88 |
|
sl@0
|
89 |
@internalComponent
|
sl@0
|
90 |
*/
|
sl@0
|
91 |
static TInt CompareC1(void*, TInt aByteSize1, const void* aPtr1, TInt aByteSize2, const void* aPtr2)
|
sl@0
|
92 |
{
|
sl@0
|
93 |
return Compare(1, aByteSize1, aPtr1, aByteSize2, aPtr2);
|
sl@0
|
94 |
}
|
sl@0
|
95 |
|
sl@0
|
96 |
/**
|
sl@0
|
97 |
User defined function which will be used by SQLITE engine for collated string comparisons,
|
sl@0
|
98 |
collation level 2.
|
sl@0
|
99 |
{aByteSize1, aPtr1} - string 1, UTF16 encoded. aByteSize1 - length in bytes.
|
sl@0
|
100 |
{aByteSize2, aPtr2} - string 2, UTF16 encoded. aByteSize2 - length in bytes.
|
sl@0
|
101 |
The function returns negative, zero or positive if the first string is less than, equal to, or
|
sl@0
|
102 |
greater than the second string.
|
sl@0
|
103 |
|
sl@0
|
104 |
@internalComponent
|
sl@0
|
105 |
*/
|
sl@0
|
106 |
static TInt CompareC2(void*, TInt aByteSize1, const void* aPtr1, TInt aByteSize2, const void* aPtr2)
|
sl@0
|
107 |
{
|
sl@0
|
108 |
return Compare(2, aByteSize1, aPtr1, aByteSize2, aPtr2);
|
sl@0
|
109 |
}
|
sl@0
|
110 |
|
sl@0
|
111 |
/**
|
sl@0
|
112 |
User defined function which will be used by SQLITE engine for collated string comparisons,
|
sl@0
|
113 |
collation level 3.
|
sl@0
|
114 |
{aByteSize1, aPtr1} - string 1, UTF16 encoded. aByteSize1 - length in bytes.
|
sl@0
|
115 |
{aByteSize2, aPtr2} - string 2, UTF16 encoded. aByteSize2 - length in bytes.
|
sl@0
|
116 |
The function returns negative, zero or positive if the first string is less than, equal to, or
|
sl@0
|
117 |
greater than the second string.
|
sl@0
|
118 |
|
sl@0
|
119 |
@internalComponent
|
sl@0
|
120 |
*/
|
sl@0
|
121 |
static TInt CompareC3(void*, TInt aByteSize1, const void* aPtr1, TInt aByteSize2, const void* aPtr2)
|
sl@0
|
122 |
{
|
sl@0
|
123 |
return Compare(3, aByteSize1, aPtr1, aByteSize2, aPtr2);
|
sl@0
|
124 |
}
|
sl@0
|
125 |
|
sl@0
|
126 |
/**
|
sl@0
|
127 |
User defined function which will be used by SQLITE engine for folded string comparisons,
|
sl@0
|
128 |
{aByteSize1, aPtr1} - string 1, UTF16 encoded. aByteSize1 - length in bytes.
|
sl@0
|
129 |
{aByteSize2, aPtr2} - string 2, UTF16 encoded. aByteSize2 - length in bytes.
|
sl@0
|
130 |
The function returns negative, zero or positive if the first string is less than, equal to, or
|
sl@0
|
131 |
greater than the second string.
|
sl@0
|
132 |
|
sl@0
|
133 |
@internalComponent
|
sl@0
|
134 |
*/
|
sl@0
|
135 |
static TInt CompareF(void*, TInt aByteSize1, const void* aPtr1, TInt aByteSize2, const void* aPtr2)
|
sl@0
|
136 |
{
|
sl@0
|
137 |
return Compare(-1, aByteSize1, aPtr1, aByteSize2, aPtr2);
|
sl@0
|
138 |
}
|
sl@0
|
139 |
|
sl@0
|
140 |
//KCollationMethodPtr array contains function pointers to all user-defined collation functions
|
sl@0
|
141 |
const TCollationMethodPtr KCollationMethodPtr[KCollationMethodCount] =
|
sl@0
|
142 |
{
|
sl@0
|
143 |
&CompareC0, &CompareC1, &CompareC2, &CompareC3, &CompareF
|
sl@0
|
144 |
};
|
sl@0
|
145 |
|
sl@0
|
146 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
147 |
///////////////////////////// TSqlCollationUtil class definition //////////////////////////////////
|
sl@0
|
148 |
///////////////////////////////////////////////////////////////////////////////////////////////////////////////
|
sl@0
|
149 |
|
sl@0
|
150 |
/**
|
sl@0
|
151 |
Initializes TSqlCollationUtil data members.
|
sl@0
|
152 |
|
sl@0
|
153 |
@panic SqlDb 4 In _DEBUG mode. aDbHandle is NULL.
|
sl@0
|
154 |
*/
|
sl@0
|
155 |
TSqlCollationUtil::TSqlCollationUtil(sqlite3* aDbHandle) :
|
sl@0
|
156 |
iDbHandle(aDbHandle)
|
sl@0
|
157 |
{
|
sl@0
|
158 |
__ASSERT_DEBUG(iDbHandle != NULL, __SQLPANIC(ESqlPanicBadArgument));
|
sl@0
|
159 |
}
|
sl@0
|
160 |
|
sl@0
|
161 |
/**
|
sl@0
|
162 |
Installs user defined collations.
|
sl@0
|
163 |
|
sl@0
|
164 |
At the moment 5 user defined collations with the following names are installed:
|
sl@0
|
165 |
- CompareC0 - 16-bit strings collated comaprison at level 0;
|
sl@0
|
166 |
- CompareC1 - 16-bit strings collated comaprison at level 1;
|
sl@0
|
167 |
- CompareC2 - 16-bit strings collated comaprison at level 2;
|
sl@0
|
168 |
- CompareC3 - 16-bit strings collated comaprison at level 3;
|
sl@0
|
169 |
- CompareF - 16-bit strings folded comaprison;
|
sl@0
|
170 |
|
sl@0
|
171 |
These user defined collations can be used in the following cases:
|
sl@0
|
172 |
|
sl@0
|
173 |
- as column constraint in "CREATE TABLE" SQL statements. For example:
|
sl@0
|
174 |
@code
|
sl@0
|
175 |
CREATE TABLE A(Col1 TEXT COLLATE CompareC1)
|
sl@0
|
176 |
@endcode
|
sl@0
|
177 |
In this case every time when Col1 values have to be compared, the SQL server will use CompareC1 collation.
|
sl@0
|
178 |
|
sl@0
|
179 |
- as column constraint in "CREATE INDEX" SQL statements. For example:
|
sl@0
|
180 |
@code
|
sl@0
|
181 |
CREATE INDEX I ON A(Col1 COLLATE CompareC2)
|
sl@0
|
182 |
@endcode
|
sl@0
|
183 |
In this case SQL server will use CompareC2 collation to compare Col1 values when using the index.
|
sl@0
|
184 |
|
sl@0
|
185 |
- In "ORDER BY" clause of "SELECT" SQL statements. For example:
|
sl@0
|
186 |
@code
|
sl@0
|
187 |
SELECT * FROM A ORDER BY Col1 COLLATE CompareC3
|
sl@0
|
188 |
@endcode
|
sl@0
|
189 |
In this case SQL server will use CompareC3 collation to compare Col1 values when building the result set.
|
sl@0
|
190 |
|
sl@0
|
191 |
@leave The function may leave with some database specific errors categorised as ESqlDbError.
|
sl@0
|
192 |
|
sl@0
|
193 |
@panic SqlDb 2 In _DEBUG mode. iDbHandle is NULL.
|
sl@0
|
194 |
*/
|
sl@0
|
195 |
void TSqlCollationUtil::InstallCollationsL()
|
sl@0
|
196 |
{
|
sl@0
|
197 |
__ASSERT_DEBUG(iDbHandle != NULL, __SQLPANIC(ESqlPanicInvalidObj));
|
sl@0
|
198 |
(void)sqlite3SymbianLastOsError();//clear last OS error
|
sl@0
|
199 |
//Register user defined collations
|
sl@0
|
200 |
for(TInt i=0;i<KCollationMethodCount;++i)
|
sl@0
|
201 |
{
|
sl@0
|
202 |
TInt err = sqlite3_create_collation16(iDbHandle, reinterpret_cast <const char*> (KCollationMethodName[i]),
|
sl@0
|
203 |
SQLITE_UTF16 | SQLITE_UTF16_ALIGNED, 0, KCollationMethodPtr[i]);
|
sl@0
|
204 |
__SQLLEAVE_IF_ERROR(::Sql2OsErrCode(err, sqlite3SymbianLastOsError()));
|
sl@0
|
205 |
}
|
sl@0
|
206 |
}
|
sl@0
|
207 |
|
sl@0
|
208 |
/**
|
sl@0
|
209 |
@return User-defined collation methods count.
|
sl@0
|
210 |
|
sl@0
|
211 |
@see TSqlCollationUtil::CollationName()
|
sl@0
|
212 |
*/
|
sl@0
|
213 |
TInt TSqlCollationUtil::CollationCount() const
|
sl@0
|
214 |
{
|
sl@0
|
215 |
return KCollationMethodCount;
|
sl@0
|
216 |
}
|
sl@0
|
217 |
|
sl@0
|
218 |
/**
|
sl@0
|
219 |
@return The name of the collation method, identified by aIndex.
|
sl@0
|
220 |
|
sl@0
|
221 |
@panic SqlDb 4 In _DEBUG mode. aIndex >= CollationMethodCount().
|
sl@0
|
222 |
|
sl@0
|
223 |
@see TSqlCollationUtil::CollationsCount()
|
sl@0
|
224 |
*/
|
sl@0
|
225 |
TPtrC TSqlCollationUtil::CollationName(TInt aIndex) const
|
sl@0
|
226 |
{
|
sl@0
|
227 |
__ASSERT_DEBUG((TUint)aIndex < KCollationMethodCount, __SQLPANIC(ESqlPanicBadArgument));
|
sl@0
|
228 |
return TPtrC(KCollationMethodName[aIndex]);
|
sl@0
|
229 |
}
|