sl@0
|
1 |
// Copyright (c) 1998-2009 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 "UT_STD.H"
|
sl@0
|
17 |
|
sl@0
|
18 |
#define UNUSED_VAR(a) a = a
|
sl@0
|
19 |
|
sl@0
|
20 |
const TUint KTableExpiry=0x100000; // ~1.0s
|
sl@0
|
21 |
|
sl@0
|
22 |
// Class Blob cleanup
|
sl@0
|
23 |
|
sl@0
|
24 |
NONSHARABLE_CLASS(CDbBlobCleanup) : public CArrayFixFlat<TDbBlobId>
|
sl@0
|
25 |
{
|
sl@0
|
26 |
public:
|
sl@0
|
27 |
static CDbBlobCleanup* NewLC(CDbBlobSpace& aBlobSpace);
|
sl@0
|
28 |
~CDbBlobCleanup();
|
sl@0
|
29 |
private:
|
sl@0
|
30 |
inline CDbBlobCleanup(CDbBlobSpace& aBlobSpace);
|
sl@0
|
31 |
private:
|
sl@0
|
32 |
CDbBlobSpace& iBlobSpace;
|
sl@0
|
33 |
};
|
sl@0
|
34 |
|
sl@0
|
35 |
inline CDbBlobCleanup::CDbBlobCleanup(CDbBlobSpace& aBlobSpace)
|
sl@0
|
36 |
: CArrayFixFlat<TDbBlobId>(8),iBlobSpace(aBlobSpace)
|
sl@0
|
37 |
{}
|
sl@0
|
38 |
|
sl@0
|
39 |
CDbBlobCleanup* CDbBlobCleanup::NewLC(CDbBlobSpace& aBlobSpace)
|
sl@0
|
40 |
{
|
sl@0
|
41 |
CDbBlobCleanup* self=new(ELeave) CDbBlobCleanup(aBlobSpace);
|
sl@0
|
42 |
CleanupStack::PushL(self);
|
sl@0
|
43 |
return self;
|
sl@0
|
44 |
}
|
sl@0
|
45 |
|
sl@0
|
46 |
CDbBlobCleanup::~CDbBlobCleanup()
|
sl@0
|
47 |
{
|
sl@0
|
48 |
TInt count=Count();
|
sl@0
|
49 |
if (count)
|
sl@0
|
50 |
{
|
sl@0
|
51 |
const TDbBlobId* blob=&(*this)[0];
|
sl@0
|
52 |
const TDbBlobId* const end=blob+count;
|
sl@0
|
53 |
for (;blob<end;++blob)
|
sl@0
|
54 |
iBlobSpace.Delete(*blob);
|
sl@0
|
55 |
}
|
sl@0
|
56 |
}
|
sl@0
|
57 |
|
sl@0
|
58 |
// Class CDbTable
|
sl@0
|
59 |
|
sl@0
|
60 |
|
sl@0
|
61 |
EXPORT_C CDbTable::CDbTable(CDbTableDatabase& aDatabase,const CDbTableDef& aDef)
|
sl@0
|
62 |
: iDatabase(&aDatabase),iDef(&aDef)
|
sl@0
|
63 |
{
|
sl@0
|
64 |
aDatabase.Open();
|
sl@0
|
65 |
aDatabase.AddTable(*this); // we reference database
|
sl@0
|
66 |
}
|
sl@0
|
67 |
|
sl@0
|
68 |
EXPORT_C CDbTable::~CDbTable()
|
sl@0
|
69 |
//
|
sl@0
|
70 |
// Destroy components
|
sl@0
|
71 |
//
|
sl@0
|
72 |
{
|
sl@0
|
73 |
__ASSERT(!InUse()); // cannot be directly deleted
|
sl@0
|
74 |
if (IsActive())
|
sl@0
|
75 |
Disconnect();
|
sl@0
|
76 |
}
|
sl@0
|
77 |
|
sl@0
|
78 |
void CDbTable::Disconnect()
|
sl@0
|
79 |
//
|
sl@0
|
80 |
// Disconnect the table from the database collection.
|
sl@0
|
81 |
//
|
sl@0
|
82 |
{
|
sl@0
|
83 |
__ASSERT(IsActive());
|
sl@0
|
84 |
Database().RemoveTable(*this);
|
sl@0
|
85 |
TRAPD(errCode, ApplyToComponentsL(CDbRecordBase::DoDelete));
|
sl@0
|
86 |
UNUSED_VAR(errCode);
|
sl@0
|
87 |
}
|
sl@0
|
88 |
|
sl@0
|
89 |
void CDbTable::Open()
|
sl@0
|
90 |
{
|
sl@0
|
91 |
__ASSERT(IsActive());
|
sl@0
|
92 |
TInt r=++iRef;
|
sl@0
|
93 |
if (r<=0)
|
sl@0
|
94 |
{ // were idle or cached
|
sl@0
|
95 |
if (r<0)
|
sl@0
|
96 |
{
|
sl@0
|
97 |
Cache().Release(*this); // was cached
|
sl@0
|
98 |
iRef=0;
|
sl@0
|
99 |
}
|
sl@0
|
100 |
Database().Open();
|
sl@0
|
101 |
}
|
sl@0
|
102 |
}
|
sl@0
|
103 |
|
sl@0
|
104 |
void CDbTable::Close()
|
sl@0
|
105 |
//
|
sl@0
|
106 |
// We may destroy this object when the last reference goes away
|
sl@0
|
107 |
//
|
sl@0
|
108 |
{
|
sl@0
|
109 |
__ASSERT(InUse());
|
sl@0
|
110 |
if (--iRef<0)
|
sl@0
|
111 |
{
|
sl@0
|
112 |
if (!IsActive())
|
sl@0
|
113 |
delete this; // disconnected table
|
sl@0
|
114 |
else
|
sl@0
|
115 |
{
|
sl@0
|
116 |
CDbTableDatabase& db=Database();
|
sl@0
|
117 |
if (!db.Transaction().IsLocked())
|
sl@0
|
118 |
Idle(); // no transaction, idle now
|
sl@0
|
119 |
db.Close(); // this must be done last to avoid early self destruction
|
sl@0
|
120 |
}
|
sl@0
|
121 |
}
|
sl@0
|
122 |
}
|
sl@0
|
123 |
|
sl@0
|
124 |
void CDbTable::Idle()
|
sl@0
|
125 |
//
|
sl@0
|
126 |
// Called when idle, change to cached state
|
sl@0
|
127 |
//
|
sl@0
|
128 |
{
|
sl@0
|
129 |
__ASSERT(IsIdle());
|
sl@0
|
130 |
__ASSERT(IsActive());
|
sl@0
|
131 |
//
|
sl@0
|
132 |
iRef=ECached;
|
sl@0
|
133 |
Cache().Hold(this,KTableExpiry); // may delete this
|
sl@0
|
134 |
}
|
sl@0
|
135 |
|
sl@0
|
136 |
void CDbTable::FlushL()
|
sl@0
|
137 |
//
|
sl@0
|
138 |
// Ensure all records objects are flushed
|
sl@0
|
139 |
//
|
sl@0
|
140 |
{
|
sl@0
|
141 |
__ASSERT(IsActive());
|
sl@0
|
142 |
if (iRef!=ECached)
|
sl@0
|
143 |
ApplyToComponentsL(CDbRecordBase::DoFlushL);
|
sl@0
|
144 |
}
|
sl@0
|
145 |
|
sl@0
|
146 |
void CDbTable::Abandon()
|
sl@0
|
147 |
//
|
sl@0
|
148 |
// Discard all components
|
sl@0
|
149 |
//
|
sl@0
|
150 |
{
|
sl@0
|
151 |
__ASSERT(IsActive());
|
sl@0
|
152 |
TRAPD(errCode, ApplyToComponentsL(CDbRecordBase::DoAbandon));
|
sl@0
|
153 |
UNUSED_VAR(errCode);
|
sl@0
|
154 |
iIndexesEnd=NULL; // flags indexes as abandoned
|
sl@0
|
155 |
++iGeneration;
|
sl@0
|
156 |
}
|
sl@0
|
157 |
|
sl@0
|
158 |
void CDbTable::Release()
|
sl@0
|
159 |
//
|
sl@0
|
160 |
// Release the table and all its cursors as DDL is about to begin
|
sl@0
|
161 |
//
|
sl@0
|
162 |
{
|
sl@0
|
163 |
__ASSERT(IsActive());
|
sl@0
|
164 |
switch (iRef)
|
sl@0
|
165 |
{
|
sl@0
|
166 |
case ECached:
|
sl@0
|
167 |
Cache().Release(*this);
|
sl@0
|
168 |
// fall throught to Idle
|
sl@0
|
169 |
case EIdle:
|
sl@0
|
170 |
delete this;
|
sl@0
|
171 |
break;
|
sl@0
|
172 |
default:
|
sl@0
|
173 |
__ASSERT(InUse());
|
sl@0
|
174 |
Database().Close();
|
sl@0
|
175 |
Disconnect();
|
sl@0
|
176 |
iDatabase=0; // this marks us as released
|
sl@0
|
177 |
iDef=0;
|
sl@0
|
178 |
break;
|
sl@0
|
179 |
}
|
sl@0
|
180 |
}
|
sl@0
|
181 |
|
sl@0
|
182 |
void CDbTable::ApplyToBlobsL(RDbRow& aRow,TBlobFuncL aFuncL,CDbBlobCleanup* aCleanup)
|
sl@0
|
183 |
{
|
sl@0
|
184 |
__ASSERT(Def().Columns().HasLongColumns());
|
sl@0
|
185 |
CDbBlobSpace* blobs=BlobsL();
|
sl@0
|
186 |
__ASSERT(blobs);
|
sl@0
|
187 |
TDbColNo col=1;
|
sl@0
|
188 |
HDbColumnSet::TIteratorC iter=Def().Columns().Begin();
|
sl@0
|
189 |
const HDbColumnSet::TIteratorC end=Def().Columns().End();
|
sl@0
|
190 |
do
|
sl@0
|
191 |
{
|
sl@0
|
192 |
if (!TDbCol::IsLong(iter->Type()))
|
sl@0
|
193 |
continue;
|
sl@0
|
194 |
const TDbColumnC column(aRow,col);
|
sl@0
|
195 |
if (column.IsNull())
|
sl@0
|
196 |
continue;
|
sl@0
|
197 |
aFuncL(*blobs,CONST_CAST(TDbBlob&,column.Blob()),iter->Type(),aCleanup);
|
sl@0
|
198 |
} while (++col,++iter<end);
|
sl@0
|
199 |
}
|
sl@0
|
200 |
|
sl@0
|
201 |
LOCAL_C void DuplicateBlobL(CDbBlobSpace& aBlobStore,TDbBlob& aBlob,TDbColType aType,CDbBlobCleanup* aCleanup)
|
sl@0
|
202 |
{
|
sl@0
|
203 |
__ASSERT(aCleanup);
|
sl@0
|
204 |
if (aBlob.IsInline())
|
sl@0
|
205 |
return;
|
sl@0
|
206 |
// need to duplicate blob
|
sl@0
|
207 |
RReadStream old(aBlobStore.ReadLC(aBlob.Id(),aType));
|
sl@0
|
208 |
TDbBlobId& newId=aCleanup->ExtendL();
|
sl@0
|
209 |
newId=KDbNullBlobId;
|
sl@0
|
210 |
RWriteStream dup(aBlobStore.CreateL(newId,aType));
|
sl@0
|
211 |
dup.PushL();
|
sl@0
|
212 |
dup.WriteL(old,aBlob.Size());
|
sl@0
|
213 |
dup.CommitL();
|
sl@0
|
214 |
CleanupStack::PopAndDestroy(2); // old and dup streams
|
sl@0
|
215 |
aBlob.SetId(newId); // row is writable
|
sl@0
|
216 |
}
|
sl@0
|
217 |
|
sl@0
|
218 |
void CDbTable::DuplicateBlobsL(RDbRow& aRow)
|
sl@0
|
219 |
//
|
sl@0
|
220 |
// duplicate any blobs
|
sl@0
|
221 |
//
|
sl@0
|
222 |
{
|
sl@0
|
223 |
if (!Def().Columns().HasLongColumns())
|
sl@0
|
224 |
return;
|
sl@0
|
225 |
|
sl@0
|
226 |
CDbBlobCleanup* cleaner=CDbBlobCleanup::NewLC(*BlobsL());
|
sl@0
|
227 |
ApplyToBlobsL(aRow,DuplicateBlobL,cleaner);
|
sl@0
|
228 |
cleaner->Reset();
|
sl@0
|
229 |
CleanupStack::PopAndDestroy();
|
sl@0
|
230 |
}
|
sl@0
|
231 |
|
sl@0
|
232 |
TBool CDbTable::ExistsL(TDbRecordId aRecordId)
|
sl@0
|
233 |
//
|
sl@0
|
234 |
// Check that aRecordId is good for this table
|
sl@0
|
235 |
//
|
sl@0
|
236 |
{
|
sl@0
|
237 |
__ASSERT(IsActive() && InUse());
|
sl@0
|
238 |
return RecordsL().ExistsL(aRecordId);
|
sl@0
|
239 |
}
|
sl@0
|
240 |
|
sl@0
|
241 |
void CDbTable::NewRowL(RDbRow& aRow)
|
sl@0
|
242 |
//
|
sl@0
|
243 |
// Initialise any auto-increment columns in the row
|
sl@0
|
244 |
//
|
sl@0
|
245 |
{
|
sl@0
|
246 |
const HDbColumnSet& columns=Def().Columns();
|
sl@0
|
247 |
if (!columns.HasAutoIncrement())
|
sl@0
|
248 |
return;
|
sl@0
|
249 |
|
sl@0
|
250 |
TUint value=RecordsL().AutoIncrementL();
|
sl@0
|
251 |
TDbColNo col=1;
|
sl@0
|
252 |
HDbColumnSet::TIteratorC iter=columns.Begin();
|
sl@0
|
253 |
const HDbColumnSet::TIteratorC end=columns.End();
|
sl@0
|
254 |
do
|
sl@0
|
255 |
{
|
sl@0
|
256 |
if (iter->iAttributes&TDbCol::EAutoIncrement)
|
sl@0
|
257 |
{
|
sl@0
|
258 |
// auto-increment only for integral types <=32 bits wide
|
sl@0
|
259 |
__ASSERT(iter->iType<=EDbColUint32);
|
sl@0
|
260 |
TDbColumn column(aRow,col);
|
sl@0
|
261 |
column.SetL(TUint32(value));
|
sl@0
|
262 |
}
|
sl@0
|
263 |
} while (++col,++iter<end);
|
sl@0
|
264 |
}
|
sl@0
|
265 |
|
sl@0
|
266 |
void CDbTable::ValidateL(const RDbRow& aRow)
|
sl@0
|
267 |
//
|
sl@0
|
268 |
// Ensure that the column data conforms to type size/flags etc
|
sl@0
|
269 |
//
|
sl@0
|
270 |
{
|
sl@0
|
271 |
HDbColumnSet::TIteratorC iter=Def().Columns().Begin();
|
sl@0
|
272 |
const HDbColumnSet::TIteratorC end=Def().Columns().End();
|
sl@0
|
273 |
const TDbCell* const last=aRow.Last();
|
sl@0
|
274 |
for (const TDbCell* column=aRow.First();column<last;++iter,column=column->Next())
|
sl@0
|
275 |
{
|
sl@0
|
276 |
TInt size=column->Length();
|
sl@0
|
277 |
if (size==0)
|
sl@0
|
278 |
{ // check for Null
|
sl@0
|
279 |
if (iter->iAttributes&TDbCol::ENotNull)
|
sl@0
|
280 |
{
|
sl@0
|
281 |
__LEAVE(KErrNotFound);
|
sl@0
|
282 |
return;
|
sl@0
|
283 |
}
|
sl@0
|
284 |
continue;
|
sl@0
|
285 |
}
|
sl@0
|
286 |
const TUint32* data=(const TUint32*)column->Data();
|
sl@0
|
287 |
switch (iter->iType)
|
sl@0
|
288 |
{
|
sl@0
|
289 |
case EDbColBit:
|
sl@0
|
290 |
if (*data>1)
|
sl@0
|
291 |
__LEAVE(KErrOverflow);
|
sl@0
|
292 |
break;
|
sl@0
|
293 |
case EDbColInt8:
|
sl@0
|
294 |
{
|
sl@0
|
295 |
TInt val=*data;
|
sl@0
|
296 |
if (TInt8(val)!=val)
|
sl@0
|
297 |
__LEAVE(KErrOverflow);
|
sl@0
|
298 |
}
|
sl@0
|
299 |
break;
|
sl@0
|
300 |
case EDbColInt16:
|
sl@0
|
301 |
{
|
sl@0
|
302 |
TInt val=*data;
|
sl@0
|
303 |
if (TInt16(val)!=val)
|
sl@0
|
304 |
__LEAVE(KErrOverflow);
|
sl@0
|
305 |
}
|
sl@0
|
306 |
break;
|
sl@0
|
307 |
case EDbColUint8:
|
sl@0
|
308 |
{
|
sl@0
|
309 |
TUint val=*data;
|
sl@0
|
310 |
if (TUint8(val)!=val)
|
sl@0
|
311 |
__LEAVE(KErrOverflow);
|
sl@0
|
312 |
}
|
sl@0
|
313 |
break;
|
sl@0
|
314 |
case EDbColUint16:
|
sl@0
|
315 |
{
|
sl@0
|
316 |
TUint val=*data;
|
sl@0
|
317 |
if (TUint16(val)!=val)
|
sl@0
|
318 |
__LEAVE(KErrOverflow);
|
sl@0
|
319 |
}
|
sl@0
|
320 |
break;
|
sl@0
|
321 |
case EDbColText16:
|
sl@0
|
322 |
size>>=1;
|
sl@0
|
323 |
case EDbColBinary:
|
sl@0
|
324 |
case EDbColText8:
|
sl@0
|
325 |
if (iter->iMaxLength==KDbUndefinedLength)
|
sl@0
|
326 |
break;
|
sl@0
|
327 |
if (size>iter->iMaxLength)
|
sl@0
|
328 |
__LEAVE(KErrOverflow);
|
sl@0
|
329 |
break;
|
sl@0
|
330 |
case EDbColLongBinary:
|
sl@0
|
331 |
case EDbColLongText8:
|
sl@0
|
332 |
case EDbColLongText16:
|
sl@0
|
333 |
if (iter->iMaxLength==KDbUndefinedLength)
|
sl@0
|
334 |
break;
|
sl@0
|
335 |
size=((TDbBlob*)data)->Size();
|
sl@0
|
336 |
if (size==KDbUndefinedLength)
|
sl@0
|
337 |
break;
|
sl@0
|
338 |
if (iter->iType==EDbColText16)
|
sl@0
|
339 |
size>>=1;
|
sl@0
|
340 |
if (size>iter->iMaxLength)
|
sl@0
|
341 |
__LEAVE(KErrOverflow);
|
sl@0
|
342 |
break;
|
sl@0
|
343 |
default:
|
sl@0
|
344 |
break;
|
sl@0
|
345 |
}
|
sl@0
|
346 |
}
|
sl@0
|
347 |
for (;iter<end;++iter)
|
sl@0
|
348 |
{ // check for Null
|
sl@0
|
349 |
if (iter->iAttributes&TDbCol::ENotNull)
|
sl@0
|
350 |
{
|
sl@0
|
351 |
__LEAVE(KErrNotFound);
|
sl@0
|
352 |
return;
|
sl@0
|
353 |
}
|
sl@0
|
354 |
}
|
sl@0
|
355 |
}
|
sl@0
|
356 |
|
sl@0
|
357 |
void CDbTable::ReadRowL(RDbRow& aRow,TDbRecordId aRecordId)
|
sl@0
|
358 |
//
|
sl@0
|
359 |
// Read a record from the table
|
sl@0
|
360 |
//
|
sl@0
|
361 |
{
|
sl@0
|
362 |
CopyToRowL(aRow,RecordsL().ReadL(aRecordId));
|
sl@0
|
363 |
}
|
sl@0
|
364 |
|
sl@0
|
365 |
void CDbTable::PrepareAppendL(const RDbTableRow& aRow)
|
sl@0
|
366 |
//
|
sl@0
|
367 |
// Validate a new record for appending
|
sl@0
|
368 |
//
|
sl@0
|
369 |
{
|
sl@0
|
370 |
EnsureIndexesL();
|
sl@0
|
371 |
ValidateL(aRow);
|
sl@0
|
372 |
CDbRecordIndex** end=iIndexesEnd;
|
sl@0
|
373 |
for (CDbRecordIndex** pix=iIndexes;pix<end;++pix)
|
sl@0
|
374 |
{
|
sl@0
|
375 |
CDbRecordIndex& ix=**pix;
|
sl@0
|
376 |
if (ix.IsBroken())
|
sl@0
|
377 |
continue;
|
sl@0
|
378 |
if (ix.FindL(KDbNullRecordId,aRow)==CDbRecordIndex::EKeyMatch)
|
sl@0
|
379 |
__LEAVE(KErrAlreadyExists); // duplicate found
|
sl@0
|
380 |
}
|
sl@0
|
381 |
}
|
sl@0
|
382 |
|
sl@0
|
383 |
TDbRecordId CDbTable::AppendRowL(const RDbTableRow& aRow)
|
sl@0
|
384 |
//
|
sl@0
|
385 |
// Validate and add a new record to the table and any open indexes
|
sl@0
|
386 |
//
|
sl@0
|
387 |
{
|
sl@0
|
388 |
CDbRecordSpace& records=RecordsL();
|
sl@0
|
389 |
CopyFromRow(records.NewL(RecordLength(aRow)),aRow);
|
sl@0
|
390 |
TDbRecordId id=records.AppendL();
|
sl@0
|
391 |
CDbRecordIndex** end=iIndexesEnd;
|
sl@0
|
392 |
for (CDbRecordIndex** pix=iIndexes;pix<end;++pix)
|
sl@0
|
393 |
{
|
sl@0
|
394 |
CDbRecordIndex& ix=**pix;
|
sl@0
|
395 |
if (ix.IsBroken())
|
sl@0
|
396 |
continue;
|
sl@0
|
397 |
__DEBUG(TInt dbgchk=) ix.InsertL(id,aRow);
|
sl@0
|
398 |
__ASSERT(dbgchk);
|
sl@0
|
399 |
}
|
sl@0
|
400 |
++iGeneration;
|
sl@0
|
401 |
return id;
|
sl@0
|
402 |
}
|
sl@0
|
403 |
|
sl@0
|
404 |
void CDbTable::PrepareReplaceL(const RDbTableRow& aRow,TDbRecordId aRecordId)
|
sl@0
|
405 |
//
|
sl@0
|
406 |
// Validate a record for replacement
|
sl@0
|
407 |
//
|
sl@0
|
408 |
{
|
sl@0
|
409 |
EnsureIndexesL();
|
sl@0
|
410 |
ValidateL(aRow);
|
sl@0
|
411 |
TUint32 update=0;
|
sl@0
|
412 |
CDbRecordIndex** end=iIndexes;
|
sl@0
|
413 |
for (CDbRecordIndex** pix=iIndexesEnd;--pix>=end;)
|
sl@0
|
414 |
{
|
sl@0
|
415 |
update<<=1;
|
sl@0
|
416 |
CDbRecordIndex& ix=**pix;
|
sl@0
|
417 |
if (ix.IsBroken())
|
sl@0
|
418 |
continue;
|
sl@0
|
419 |
switch (ix.FindL(aRecordId,aRow))
|
sl@0
|
420 |
{
|
sl@0
|
421 |
case CDbRecordIndex::ENoMatch: // key has changed in index
|
sl@0
|
422 |
update|=1;
|
sl@0
|
423 |
break;
|
sl@0
|
424 |
case CDbRecordIndex::EKeyMatch: // duplicate found
|
sl@0
|
425 |
__LEAVE(KErrAlreadyExists);
|
sl@0
|
426 |
case CDbRecordIndex::EEntryMatch: // no change in index
|
sl@0
|
427 |
break;
|
sl@0
|
428 |
}
|
sl@0
|
429 |
}
|
sl@0
|
430 |
iUpdateMap=update;
|
sl@0
|
431 |
}
|
sl@0
|
432 |
|
sl@0
|
433 |
void CDbTable::DoReplaceRowL(const RDbRow& aRow,TDbRecordId aRecordId)
|
sl@0
|
434 |
{
|
sl@0
|
435 |
CopyFromRow(RecordsL().ReplaceL(aRecordId,RecordLength(aRow)),aRow);
|
sl@0
|
436 |
++iGeneration;
|
sl@0
|
437 |
}
|
sl@0
|
438 |
|
sl@0
|
439 |
void CDbTable::ReplaceRowL(RDbTableRow& aRow,TDbRecordId aRecordId)
|
sl@0
|
440 |
//
|
sl@0
|
441 |
// Replace a record in the table
|
sl@0
|
442 |
//
|
sl@0
|
443 |
{
|
sl@0
|
444 |
if (Def().Columns().HasLongColumns())
|
sl@0
|
445 |
CheckInliningL(aRow);
|
sl@0
|
446 |
TUint32 update=iUpdateMap;
|
sl@0
|
447 |
if (update==0)
|
sl@0
|
448 |
{
|
sl@0
|
449 |
DoReplaceRowL(aRow,aRecordId);
|
sl@0
|
450 |
return;
|
sl@0
|
451 |
}
|
sl@0
|
452 |
RDbTableRow oldRow; // temporary row buffer for old row values
|
sl@0
|
453 |
oldRow.Open(this);
|
sl@0
|
454 |
oldRow.PushL(); // cleanup buffer if there is trouble
|
sl@0
|
455 |
ReadRowL(oldRow,aRecordId);
|
sl@0
|
456 |
DoReplaceRowL(aRow,aRecordId);
|
sl@0
|
457 |
for (CDbRecordIndex** pix=iIndexes;update;++pix,update>>=1)
|
sl@0
|
458 |
{
|
sl@0
|
459 |
if (update&1)
|
sl@0
|
460 |
{
|
sl@0
|
461 |
CDbRecordIndex& index=**pix;
|
sl@0
|
462 |
index.DeleteL(aRecordId,oldRow);
|
sl@0
|
463 |
__DEBUG(TInt dbgchk=) index.InsertL(aRecordId,aRow);
|
sl@0
|
464 |
__ASSERT(dbgchk);
|
sl@0
|
465 |
}
|
sl@0
|
466 |
}
|
sl@0
|
467 |
CleanupStack::PopAndDestroy(); // temp row buffer
|
sl@0
|
468 |
}
|
sl@0
|
469 |
|
sl@0
|
470 |
LOCAL_C void CheckInlineL(CDbBlobSpace& aBlobStore,TDbBlob& aBlob,TDbColType aType,CDbBlobCleanup*)
|
sl@0
|
471 |
{
|
sl@0
|
472 |
if (!aBlob.IsInline())
|
sl@0
|
473 |
return;
|
sl@0
|
474 |
if (aBlob.Size()>aBlobStore.InlineLimit())
|
sl@0
|
475 |
aBlob.SetId(aBlobStore.CreateL(aType,aBlob.Data(),aBlob.Size()));
|
sl@0
|
476 |
}
|
sl@0
|
477 |
|
sl@0
|
478 |
void CDbTable::CheckInliningL(RDbRow& aRow)
|
sl@0
|
479 |
//
|
sl@0
|
480 |
// Ensure that all Blobs are within the current inline limit
|
sl@0
|
481 |
//
|
sl@0
|
482 |
{
|
sl@0
|
483 |
ApplyToBlobsL(aRow,CheckInlineL);
|
sl@0
|
484 |
}
|
sl@0
|
485 |
|
sl@0
|
486 |
LOCAL_C void DiscardBlobL(CDbBlobSpace& aBlobStore,TDbBlob& aBlob,TDbColType,CDbBlobCleanup*)
|
sl@0
|
487 |
{
|
sl@0
|
488 |
if (!aBlob.IsInline())
|
sl@0
|
489 |
aBlobStore.DeleteL(aBlob.Id());
|
sl@0
|
490 |
}
|
sl@0
|
491 |
|
sl@0
|
492 |
EXPORT_C void CDbTable::DiscardBlobsL(RDbRow& aRow)
|
sl@0
|
493 |
//
|
sl@0
|
494 |
// Default implemtation xlates the record and then walks the row buffer
|
sl@0
|
495 |
//
|
sl@0
|
496 |
{
|
sl@0
|
497 |
ApplyToBlobsL(aRow,DiscardBlobL);
|
sl@0
|
498 |
}
|
sl@0
|
499 |
|
sl@0
|
500 |
void CDbTable::DeleteRowL(RDbTableRow& aRow,TDbRecordId aRecordId)
|
sl@0
|
501 |
//
|
sl@0
|
502 |
// Delete the record from the file and unlock it.
|
sl@0
|
503 |
//
|
sl@0
|
504 |
{
|
sl@0
|
505 |
EnsureIndexesL();
|
sl@0
|
506 |
|
sl@0
|
507 |
if (Def().Columns().HasLongColumns())
|
sl@0
|
508 |
{
|
sl@0
|
509 |
// Read data from the stream but do not delete the stream yet.
|
sl@0
|
510 |
aRow.ReadL(aRecordId);
|
sl@0
|
511 |
}
|
sl@0
|
512 |
|
sl@0
|
513 |
CDbRecordIndex** end=iIndexes;
|
sl@0
|
514 |
CDbRecordIndex** pix=iIndexesEnd;
|
sl@0
|
515 |
if (pix!=end)
|
sl@0
|
516 |
aRow.ReadL(aRecordId);
|
sl@0
|
517 |
RecordsL().EraseL(aRecordId);
|
sl@0
|
518 |
while (--pix>=end)
|
sl@0
|
519 |
{
|
sl@0
|
520 |
CDbRecordIndex& ix=**pix;
|
sl@0
|
521 |
if (!ix.IsBroken())
|
sl@0
|
522 |
ix.DeleteL(aRecordId,aRow);
|
sl@0
|
523 |
}
|
sl@0
|
524 |
|
sl@0
|
525 |
if (Def().Columns().HasLongColumns())
|
sl@0
|
526 |
{
|
sl@0
|
527 |
// Now delete the stream.
|
sl@0
|
528 |
DiscardBlobsL(aRow);
|
sl@0
|
529 |
}
|
sl@0
|
530 |
|
sl@0
|
531 |
++iGeneration;
|
sl@0
|
532 |
}
|
sl@0
|
533 |
|
sl@0
|
534 |
EXPORT_C CDbRecordSpace& CDbTable::RecordsL()
|
sl@0
|
535 |
{
|
sl@0
|
536 |
__ASSERT(IsActive() && InUse());
|
sl@0
|
537 |
CDbRecordSpace* rec=iRecords;
|
sl@0
|
538 |
if (rec==NULL)
|
sl@0
|
539 |
iRecords=rec=RecordSpaceL();
|
sl@0
|
540 |
if (rec->OpenL())
|
sl@0
|
541 |
__LEAVE(KErrCorrupt);
|
sl@0
|
542 |
return *rec;
|
sl@0
|
543 |
}
|
sl@0
|
544 |
|
sl@0
|
545 |
EXPORT_C CDbBlobSpace* CDbTable::BlobsL()
|
sl@0
|
546 |
{
|
sl@0
|
547 |
__ASSERT(IsActive() && InUse());
|
sl@0
|
548 |
CDbBlobSpace* blob=iBlobs;
|
sl@0
|
549 |
if (blob==NULL)
|
sl@0
|
550 |
iBlobs=blob=BlobSpaceL();
|
sl@0
|
551 |
if (blob->OpenL())
|
sl@0
|
552 |
__LEAVE(KErrCorrupt);
|
sl@0
|
553 |
return blob;
|
sl@0
|
554 |
}
|
sl@0
|
555 |
|
sl@0
|
556 |
EXPORT_C CDbRecordIndex& CDbTable::IndexL(const CDbTableIndexDef& aIndex)
|
sl@0
|
557 |
//
|
sl@0
|
558 |
// Load the index associated with the index definition and ensure it is operational
|
sl@0
|
559 |
//
|
sl@0
|
560 |
{
|
sl@0
|
561 |
__ASSERT(IsActive() && InUse());
|
sl@0
|
562 |
// find the matching slot in the indexes array
|
sl@0
|
563 |
CDbRecordIndex** slot=&iIndexes[0];
|
sl@0
|
564 |
for (TSglQueIterC<CDbTableIndexDef> iter(Def().Indexes().AsQue());iter++!=&aIndex;)
|
sl@0
|
565 |
++slot;
|
sl@0
|
566 |
__ASSERT(iIndexesEnd==NULL||(slot>=iIndexes&&slot<iIndexesEnd));
|
sl@0
|
567 |
// load (if required) and open the index
|
sl@0
|
568 |
CDbRecordIndex* index=*slot;
|
sl@0
|
569 |
if (index==0)
|
sl@0
|
570 |
*slot=index=RecordIndexL(aIndex);
|
sl@0
|
571 |
if (index->OpenL())
|
sl@0
|
572 |
__LEAVE(KErrCorrupt);
|
sl@0
|
573 |
return *index;
|
sl@0
|
574 |
}
|
sl@0
|
575 |
|
sl@0
|
576 |
void CDbTable::EnsureIndexesL()
|
sl@0
|
577 |
//
|
sl@0
|
578 |
// Ensure that all indexes are open
|
sl@0
|
579 |
//
|
sl@0
|
580 |
{
|
sl@0
|
581 |
__ASSERT(IsActive() && InUse());
|
sl@0
|
582 |
if (iIndexesEnd==NULL)
|
sl@0
|
583 |
{
|
sl@0
|
584 |
CDbRecordIndex** pp=iIndexes;
|
sl@0
|
585 |
TSglQueIterC<CDbTableIndexDef> iter(Def().Indexes().AsQue());
|
sl@0
|
586 |
for (const CDbTableIndexDef* xDef;(xDef=iter++)!=NULL;++pp)
|
sl@0
|
587 |
{
|
sl@0
|
588 |
CDbRecordIndex* ix=*pp;
|
sl@0
|
589 |
if (ix==NULL)
|
sl@0
|
590 |
*pp=ix=RecordIndexL(*xDef);
|
sl@0
|
591 |
ix->OpenL(); // ignore broken-ness
|
sl@0
|
592 |
}
|
sl@0
|
593 |
iIndexesEnd=pp;
|
sl@0
|
594 |
}
|
sl@0
|
595 |
}
|
sl@0
|
596 |
|
sl@0
|
597 |
CDbRecordIter* CDbTable::IteratorL()
|
sl@0
|
598 |
{
|
sl@0
|
599 |
return RecordsL().IteratorL();
|
sl@0
|
600 |
}
|
sl@0
|
601 |
|
sl@0
|
602 |
CDbRecordIter* CDbTable::IteratorL(const CDbTableIndexDef& aIndex,TUint aInclusion,const TDbLookupKey* aLowerBound,const TDbLookupKey* aUpperBound)
|
sl@0
|
603 |
//
|
sl@0
|
604 |
// create an interator for the index parameter
|
sl@0
|
605 |
//
|
sl@0
|
606 |
{
|
sl@0
|
607 |
return IndexL(aIndex).IteratorL(aInclusion,aLowerBound,aUpperBound);
|
sl@0
|
608 |
}
|
sl@0
|
609 |
|
sl@0
|
610 |
EXPORT_C TInt CDbTable::IndexSpanL(const CDbTableIndexDef&,TUint,const TDbLookupKey*,const TDbLookupKey*)
|
sl@0
|
611 |
//
|
sl@0
|
612 |
// Default implementation: no statistics are available
|
sl@0
|
613 |
//
|
sl@0
|
614 |
{
|
sl@0
|
615 |
return EUnavailableSpan;
|
sl@0
|
616 |
}
|
sl@0
|
617 |
|
sl@0
|
618 |
CDbRecordIter* CDbTable::IteratorL(const TDesC& aIndex)
|
sl@0
|
619 |
//
|
sl@0
|
620 |
// create an interator for the index named
|
sl@0
|
621 |
//
|
sl@0
|
622 |
{
|
sl@0
|
623 |
return IteratorL(Def().Indexes().FindL(aIndex));
|
sl@0
|
624 |
}
|
sl@0
|
625 |
|
sl@0
|
626 |
void CDbTable::ApplyToComponentsL(void (*anOperationL)(CDbRecordBase*))
|
sl@0
|
627 |
//
|
sl@0
|
628 |
// Invoke anOperation on all components of the table
|
sl@0
|
629 |
//
|
sl@0
|
630 |
{
|
sl@0
|
631 |
if (iRecords)
|
sl@0
|
632 |
anOperationL(iRecords);
|
sl@0
|
633 |
if (iBlobs)
|
sl@0
|
634 |
anOperationL(iBlobs);
|
sl@0
|
635 |
CDbRecordIndex** const ixs=iIndexes;
|
sl@0
|
636 |
CDbRecordIndex** pix=iIndexesEnd;
|
sl@0
|
637 |
if (pix==NULL)
|
sl@0
|
638 |
pix=&iIndexes[KDbTableMaxIndexes];
|
sl@0
|
639 |
while (--pix>=ixs)
|
sl@0
|
640 |
if (*pix)
|
sl@0
|
641 |
anOperationL(*pix);
|
sl@0
|
642 |
}
|
sl@0
|
643 |
|
sl@0
|
644 |
// Class CDbtable::TValid
|
sl@0
|
645 |
|
sl@0
|
646 |
// this class is used by the cursor to check that it is still operational
|
sl@0
|
647 |
|
sl@0
|
648 |
CDbTable::TValid::TValid(CDbTable& aTable)
|
sl@0
|
649 |
:iTable(aTable)
|
sl@0
|
650 |
{
|
sl@0
|
651 |
__ASSERT(aTable.IsActive());
|
sl@0
|
652 |
iRollback.Construct(aTable.Database().Transaction().RollbackGeneration());
|
sl@0
|
653 |
}
|
sl@0
|
654 |
|
sl@0
|
655 |
TBool CDbTable::TValid::Reset()
|
sl@0
|
656 |
{
|
sl@0
|
657 |
TBool b=Table().IsActive();
|
sl@0
|
658 |
if (b)
|
sl@0
|
659 |
iRollback.Mark();
|
sl@0
|
660 |
return b;
|
sl@0
|
661 |
}
|
sl@0
|
662 |
|
sl@0
|
663 |
void CDbTable::TValid::CheckL() const
|
sl@0
|
664 |
{
|
sl@0
|
665 |
CDbTableDatabase* d=Table().iDatabase;
|
sl@0
|
666 |
if (!d)
|
sl@0
|
667 |
__LEAVE(KErrDisconnected);
|
sl@0
|
668 |
else
|
sl@0
|
669 |
d->Transaction().ReadyL();
|
sl@0
|
670 |
if (iRollback.Changed())
|
sl@0
|
671 |
__LEAVE(KErrNotReady);
|
sl@0
|
672 |
}
|