Update contrib.
1 // Copyright (c) 1998-2009 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.
14 // DBMS server: data source management and sharing classes
20 extern const TDbDriver KBuiltinDriver;
22 // Class CDbsObserver::HObserver
24 // The server side of a RDbNotifier object
25 // All "observers" on the same database are held in a list on the
26 // primary CDbsObserver, which tracks the single notifier object
27 // on the data source.
29 inline CDbsObserver::HObserver::HObserver()
34 // Complete the outstanding request, and reset the observer status
36 void CDbsObserver::HObserver::Complete(TInt aStatus)
39 iMessage.Complete(aStatus);
43 // Notification request from the client
44 // Int0() has the notification type (CDbNotifier::TType)
46 void CDbsObserver::HObserver::Notify(const RMessage2& aMessage)
48 __ASSERT_ALWAYS(iPending>=0,Panic(EDbsObserverRequestPending));
50 if (iPending>RDbNotifier::EUnlock)
51 Complete(iPending); // report any missed event first
52 else if (iLink.iNext==&iLink)
53 Complete(RDbNotifier::EClose); // report a "closed" event
55 iPending=aMessage.Int0(); // wait for an event
59 // Cancel the notification request (if pending)
61 void CDbsObserver::HObserver::Cancel()
68 // An event occurs on the database
70 void CDbsObserver::HObserver::Event(TInt aEvent)
72 if (aEvent==RDbNotifier::EClose)
73 { // detach the observer when closed
75 iLink.iPrev=iLink.iNext=&iLink;
77 TInt pending=iPending;
79 { // request is pending
80 if (aEvent==RDbNotifier::EUnlock && pending==CDbNotifier::EChange)
81 ; // don't report unlock events to "change" requests
85 else if (aEvent>pending)
86 iPending=aEvent; // store more significant event
90 // Client notifer is closed
92 CDbsObserver::HObserver::~HObserver()
100 // The central server-side observer active object for the database notifiers
101 // This maintains a list of all notifiers, and propogates events from the
104 inline CDbsObserver::CDbsObserver(CDbsSource& aSource)
105 :CActive(10),iSource(aSource),iQueue(_FOFF(HObserver,iLink))
108 CDbsObserver* CDbsObserver::NewL(CDbsSource& aSource)
110 CDbsObserver* self=new(ELeave) CDbsObserver(aSource);
111 CleanupStack::PushL(self);
112 self->iNotifier=aSource.Source().NotifierL();
113 CActiveScheduler::Add(self);
114 CleanupStack::Pop(); // self
119 // Used by the source to destroy the observer only when all client
120 // notifiers have been closed
122 CDbsObserver* CDbsObserver::Collect(CDbsObserver* aObserver)
126 if (!aObserver->iQueue.IsEmpty())
132 CDbsObserver::~CDbsObserver()
134 __ASSERT(iQueue.IsEmpty());
135 CDbObject::Destroy(iNotifier); // will cancel any request
140 // Create and initialise a new client-observer object
142 CDbsObserver::HObserver* CDbsObserver::ObserverL()
144 HObserver* observer=new(ELeave) HObserver;
145 iQueue.AddLast(*observer);
147 Queue(); // start receiving events
152 // Request an event from the database
154 void CDbsObserver::Queue()
157 iNotifier->Notify(CDbNotifier::EUnlock,iStatus);
161 // Dispatch the event to all observers and re-queue
163 void CDbsObserver::RunL()
165 TDblQueIter<HObserver> iter(iQueue);
166 for (HObserver* ob;(ob=iter++)!=0;)
167 ob->Event(iStatus.Int());
168 if (!iQueue.IsEmpty())
170 else if (iStatus.Int()==RDbNotifier::EClose)
171 iSource.Closed(); // disconnect and destroy on a close event
175 // Provided fo CActive: should never have to do anything as called only
176 // via the d'tor which destroys the notifier first, completing the message
178 void CDbsObserver::DoCancel()
180 __ASSERT(iStatus!=KRequestPending);
183 // Class CDbsDatabaseStub
185 // This class is used as a stub object between the two phases of
186 // authentication on a secure database
188 CDbsDatabaseStub* CDbsDatabaseStub::NewL()
190 return new(ELeave) CDbsDatabaseStub;
194 // Authenticate the access, on success destroy this object and return the
195 // real database interface object
197 CDbDatabase* CDbsDatabaseStub::AuthenticateL()
199 CDbSource& src=CDbsConnection::Source(*this).Source();
200 CDbDatabase* db=src.AuthenticateL();
206 // Class CDbsConnection
208 // The context for all interface objects residing in the server, which
209 // owns a reference on the data source.
211 CDbsConnection::~CDbsConnection()
220 // The sharing point for databases in the server, this maintains access
221 // to the CDbSource interface object, allowing multiple connections to the
225 // Something has closed. Check if we are still needed, and delete if not
227 void CDbsSource::Closed()
229 __ASSERT(iConnections==0);
230 iObserver=CDbsObserver::Collect(iObserver);
236 // A connection has been removed
238 void CDbsSource::Close()
240 __ASSERT(iConnections>0);
242 if (--iConnections==0)
243 { // last connection is closed
244 CDbSource* s=iSource;
246 CDbObject::Destroy(s);
247 iLink.Deque(); // cannot connect this source again
252 CDbsSource::~CDbsSource()
255 __ASSERT(!iObserver);
260 // Construct a new source object from a database using the driver discovery
262 CDbsSource* CDbsSource::NewL(RFs& aFs,const TDesC& aSource)
264 //The following two statements are here to check the file path and raise (if have to)
265 //the same errors as they were raised before by the previous (DbDriver related) source code.
267 __LEAVE_IF_ERROR(aFs.Entry(aSource, fileEntry));
269 const TDbFormat& fmt=KBuiltinDriver.iFormats[0];
271 CDbsSource* self=new(ELeave) CDbsSource(fmt);
272 CleanupStack::PushL(self);
273 self->iName=aSource.AllocL();
274 self->iSource=fmt.OpenL(aFs,aSource,TDbFormat::EReadWrite);
275 CleanupStack::Pop(); // self
280 // [Create and] return the active observer of the data source
282 CDbsObserver::HObserver* CDbsSource::ObserverL()
285 CDbsObserver* observer=iObserver;
287 iObserver=observer=CDbsObserver::NewL(*this);
288 return observer->ObserverL();
292 // Test to find an identical source, in order to share it
294 TBool CDbsSource::Is(const TDesC& aSource) const
296 if (iName->CompareF(aSource)!=0)
305 // The collection of all shared sources in the server
308 // Open a source for sharing
310 CDbsConnection* RDbsSources::OpenLC(RFs& aFs,const TDesC& aSource,const TDesC& /*aFormat*/)
312 CDbsConnection* connect=new(ELeave) CDbsConnection;
313 CleanupStack::PushL(connect);
315 // try and find the source already open
316 TIter iter(iSources);
322 { // not already open, have to open a new source
323 src=CDbsSource::NewL(aFs,aSource);
324 iSources.AddFirst(*src);
327 if (src->Is(aSource))
328 break; // share this source