os/persistentdata/persistentstorage/dbms/sdbms/SD_SRV.CPP
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
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
// DBMS server implementation
sl@0
    15
// 
sl@0
    16
//
sl@0
    17
sl@0
    18
#include "SD_STD.H"
sl@0
    19
#include "Sd_PlatDep.h"
sl@0
    20
sl@0
    21
using namespace DBSC;
sl@0
    22
sl@0
    23
//CDbmsActiveScheduler class just exposes the access to 
sl@0
    24
//CActiveScheduler::Level() method, which is needed to 
sl@0
    25
//make a decision when to call CDbmsActiveScheduler::Stop().
sl@0
    26
class CDbmsActiveScheduler : public CActiveScheduler
sl@0
    27
	{
sl@0
    28
public:
sl@0
    29
	inline TInt Level() const
sl@0
    30
		{
sl@0
    31
		return CActiveScheduler::Level();
sl@0
    32
		}
sl@0
    33
	};
sl@0
    34
sl@0
    35
///////////////////////
sl@0
    36
// Class CDbsServer
sl@0
    37
inline CDbsServer::CDbsServer() :
sl@0
    38
	CServer2(0, ESharableSessions), 
sl@0
    39
	iSources(iCache),
sl@0
    40
	iDbPropsFactory(iFs),
sl@0
    41
	iDriveSpaceCol(iFs)
sl@0
    42
	{
sl@0
    43
	DbgPrint1(_L("###CDbsServer::CDbsServer(), Server ProcID=%d\n"), RDbProcess().Id());
sl@0
    44
	}
sl@0
    45
sl@0
    46
//"iCache.Hold(this,KDbsExitDelay)" statement will put CDbsServer instance in the cache, 
sl@0
    47
//which ensures that CDbsServer instance will be automatically destroyed if nobody
sl@0
    48
//uses it next KDbsExitDelay microseconds.
sl@0
    49
CDbsServer* CDbsServer::NewL()
sl@0
    50
	{
sl@0
    51
	DbgPrint1(_L("###CDbsServer::NewL(), Server ProcId=%d\n"), RDbProcess().Id());
sl@0
    52
	CDbsServer* self= new (ELeave) CDbsServer;
sl@0
    53
	CleanupStack::PushL(self);
sl@0
    54
	self->ConstructL();
sl@0
    55
	CleanupStack::Pop(self);
sl@0
    56
	self->iCache.Hold(self, KDbsExitDelay);		// the last thing we do here
sl@0
    57
	//Intentional behaviour, resources freed after timeout only
sl@0
    58
	//coverity[use_after_free]
sl@0
    59
	return self;
sl@0
    60
	}
sl@0
    61
sl@0
    62
//
sl@0
    63
// 2nd phase construction - ensure the timer and server objects are running
sl@0
    64
//
sl@0
    65
void CDbsServer::ConstructL()
sl@0
    66
	{
sl@0
    67
	DbgPrint1(_L("###CDbsServer::ConstructL(), Server ProcId=%d\n"), RDbProcess().Id());
sl@0
    68
	__LEAVE_IF_ERROR(Dll::SetTls(this));
sl@0
    69
	StartL(KDbsServerName);
sl@0
    70
	__LEAVE_IF_ERROR(iFs.Connect());
sl@0
    71
	iDbPropsFactory.OpenL();
sl@0
    72
	//EDriveZ - Symbian OS ROM drive !?!?
sl@0
    73
	iDbPropsFactory.GetPrivatePathL(EDriveZ, iFileName);
sl@0
    74
	iPolicyProxy = CPolicyProxy::NewL(iFs,iFileName);
sl@0
    75
	User::LeaveIfError(iCache.Open(ECacheSize, ETrue));	// create a timed cache
sl@0
    76
	}
sl@0
    77
sl@0
    78
//
sl@0
    79
// Invoked via cache entry expiry (or cleanup)
sl@0
    80
//
sl@0
    81
CDbsServer::~CDbsServer()
sl@0
    82
	{
sl@0
    83
	DbgPrint1(_L("###CDbsServer::~CDbsServer(), Server ProcId=%d\n"), RDbProcess().Id());
sl@0
    84
	iDriveSpaceCol.Close();
sl@0
    85
	delete iPolicyProxy;
sl@0
    86
	iDbPropsFactory.Close();
sl@0
    87
	iCache.Close();
sl@0
    88
	iSources.Close();
sl@0
    89
	iFs.Close();
sl@0
    90
	//Stop the scheduler if the nesting level > 0.
sl@0
    91
	//Sometime it happens that more than one thread tries to run the server and you will have
sl@0
    92
	//CDbmsActiveScheduler::Install() called multiple times. But you have to stop the 
sl@0
    93
	//scheduler only once!
sl@0
    94
	CDbmsActiveScheduler* scheduler = static_cast <CDbmsActiveScheduler*> (CActiveScheduler::Current());
sl@0
    95
	if(scheduler->Level() > 0)
sl@0
    96
		{
sl@0
    97
		DbgPrint2(_L("###CDbsServer::~CDbsServer(), stop the scheduler, Server ProcId=%d, scheduler=%x\n"), RDbProcess().Id(), (TInt)scheduler);
sl@0
    98
		CDbmsActiveScheduler::Stop();
sl@0
    99
		}
sl@0
   100
	}
sl@0
   101
sl@0
   102
//
sl@0
   103
// Creates a new client session. This should really check the version number.
sl@0
   104
//
sl@0
   105
CSession2* CDbsServer::NewSessionL(const TVersion& aVersion,const RMessage2&) const
sl@0
   106
	{
sl@0
   107
	if(!User::QueryVersionSupported(RDbs::Version(), aVersion))
sl@0
   108
		{
sl@0
   109
		__LEAVE(KErrNotSupported);
sl@0
   110
		}
sl@0
   111
	CSession2* session = new (ELeave) CDbsSession;
sl@0
   112
	iCache.Release(*this);
sl@0
   113
	return session;
sl@0
   114
	}
sl@0
   115
sl@0
   116
//
sl@0
   117
// Returns the instance of the server
sl@0
   118
//
sl@0
   119
CDbsServer* CDbsServer::Instance()
sl@0
   120
	{
sl@0
   121
	return (CDbsServer*)Dll::Tls();
sl@0
   122
	}
sl@0
   123
sl@0
   124
//
sl@0
   125
// initiates exit if the last client is closing
sl@0
   126
//
sl@0
   127
void CDbsServer::RemoveSession()
sl@0
   128
	{
sl@0
   129
	iSessionIter.SetToFirst();
sl@0
   130
	iSessionIter++;
sl@0
   131
	if(iSessionIter++ == 0)
sl@0
   132
		{
sl@0
   133
		iCache.Hold(this, KDbsExitDelay);
sl@0
   134
		}
sl@0
   135
	}
sl@0
   136
sl@0
   137
//
sl@0
   138
// Performs all server initialisation, in particular creation of the
sl@0
   139
// scheduler and server.
sl@0
   140
// Noting has to be leaved in the cleanup stack!
sl@0
   141
// Both: DBMS server and the active scheduler will be destroyed
sl@0
   142
// later. The active scheduler - in CDbsServer destructor.
sl@0
   143
// CDbsServer instance - by iCache instance (RDbCache).
sl@0
   144
static void CreateServerL()
sl@0
   145
	{
sl@0
   146
	DbgPrint1(_L("###DBMS-CreateServerL(), Server ProcId=%d\n"), RDbProcess().Id());
sl@0
   147
	// naming the server thread after the server helps to debug panics
sl@0
   148
	User::LeaveIfError(User::RenameThread(KDbsServerName));
sl@0
   149
	// ensure the server thread has a handle on EDBMS.DLL
sl@0
   150
	LoadDbmsLibraryL();
sl@0
   151
sl@0
   152
	// create and install the active scheduler we need
sl@0
   153
	CDbmsActiveScheduler* scheduler = new (ELeave) CDbmsActiveScheduler;
sl@0
   154
	CleanupStack::PushL(scheduler);
sl@0
   155
	CDbmsActiveScheduler::Install(scheduler);
sl@0
   156
sl@0
   157
	// create the server
sl@0
   158
	(void)CDbsServer::NewL();
sl@0
   159
sl@0
   160
	CleanupStack::Pop(scheduler);
sl@0
   161
sl@0
   162
	// Initialisation complete, now signal the client
sl@0
   163
	RDbProcess::Rendezvous(KErrNone);
sl@0
   164
	}
sl@0
   165
sl@0
   166
//
sl@0
   167
// Performs all server initialisation, in particular creation of the
sl@0
   168
// scheduler and server and then run the scheduler
sl@0
   169
//
sl@0
   170
static void RunServerL()
sl@0
   171
	{
sl@0
   172
	DbgPrint1(_L("###DBMS-RunServerL(), Server ProcId=%d\n"), RDbProcess().Id());
sl@0
   173
	::CreateServerL();
sl@0
   174
	//After successfull creation CreateServerL() call leaves the active scheduler instance
sl@0
   175
	//and the DBMS server instance in the heap - they will be destroyed later.
sl@0
   176
	__UHEAP_MARK;
sl@0
   177
	//Start the scheduler. The execution control is transferred to the curent
sl@0
   178
	//active scheduler. The statement after "CDbmsActiveScheduler::Start();" will
sl@0
   179
	//be reached after the stop of active scheduler.
sl@0
   180
	CDbmsActiveScheduler::Start();
sl@0
   181
	__UHEAP_MARKEND;
sl@0
   182
	}
sl@0
   183
sl@0
   184
//
sl@0
   185
// Server process entry-point
sl@0
   186
//
sl@0
   187
EXPORT_C TInt Dbs::Run()
sl@0
   188
	{
sl@0
   189
	CTrapCleanup* cleanup = CTrapCleanup::New();
sl@0
   190
	TInt err = KErrNoMemory;
sl@0
   191
	if(cleanup)
sl@0
   192
		{
sl@0
   193
		TRAP(err, ::RunServerL());
sl@0
   194
		delete cleanup;
sl@0
   195
		}
sl@0
   196
	DbgPrint2(_L("###Dbs::Run(), end, Server ProcId=%d, err=%d\n"), RDbProcess().Id(), err);
sl@0
   197
	return err;
sl@0
   198
	}
sl@0
   199
	
sl@0
   200
///////////////////////
sl@0
   201
// Client called code
sl@0
   202
sl@0
   203
//
sl@0
   204
// Starts the DBMS server
sl@0
   205
//
sl@0
   206
TInt Dbs::Start()
sl@0
   207
	{
sl@0
   208
	DbgPrint1(_L("#-#Dbs::Start(), Client ProcId=%d\n"), RDbProcess().Id());
sl@0
   209
	RDbProcess server;
sl@0
   210
	TInt err = ::CreateDbmsProcess(server);
sl@0
   211
	if(err != KErrNone)
sl@0
   212
		{
sl@0
   213
		DbgPrint2(_L("#-#Dbs::Start(), CreateDbmsProcess, Client ProcId=%d, err=%d\n"), RDbProcess().Id(), err);
sl@0
   214
		return err;
sl@0
   215
		}
sl@0
   216
sl@0
   217
	TRequestStatus stat;
sl@0
   218
	server.Rendezvous(stat);
sl@0
   219
	if(stat != KRequestPending)
sl@0
   220
		{
sl@0
   221
		server.Kill(0);		// abort startup
sl@0
   222
		}
sl@0
   223
	else
sl@0
   224
		{
sl@0
   225
		server.Resume();	// logon OK - start the server
sl@0
   226
		}
sl@0
   227
	User::WaitForRequest(stat);		// wait for start or death
sl@0
   228
sl@0
   229
	// we can't use the 'exit reason' if the server panicked as this
sl@0
   230
	// is the panic 'reason' and may be '0' which cannot be distinguished
sl@0
   231
	// from KErrNone
sl@0
   232
	err = (server.ExitType() == EExitPanic) ? KErrGeneral : stat.Int();
sl@0
   233
	server.Close();
sl@0
   234
	DbgPrint2(_L("#-#Dbs::Start(), end, Client ProcId=%d, err=%d\n"), RDbProcess().Id(), err);
sl@0
   235
	return err;
sl@0
   236
	}
sl@0
   237