os/kernelhwsrv/kernel/eka/include/drivers/resourcecontrol.h
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
Update contrib.
sl@0
     1
// Copyright (c) 2007-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 the License "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
// e32\include\drivers\resourcecontrol.h
sl@0
    15
// 
sl@0
    16
// WARNING: This file contains some APIs which are internal and are subject
sl@0
    17
//          to change without notice. Such APIs should therefore not be used
sl@0
    18
//          outside the Kernel and Hardware Services package.
sl@0
    19
//
sl@0
    20
sl@0
    21
#ifndef __RESOURCECONTROL_H__
sl@0
    22
#define __RESOURCECONTROL_H__
sl@0
    23
#include <nklib.h>
sl@0
    24
#include <kernel/kernel.h>
sl@0
    25
#include <kernel/kern_priv.h>
sl@0
    26
#include <e32ver.h>
sl@0
    27
#define PRM_CONTROLLER
sl@0
    28
#ifndef PRM_ENABLE_EXTENDED_VERSION
sl@0
    29
#include <drivers/resource.h>
sl@0
    30
#else
sl@0
    31
#include <drivers/resource_extend.h>
sl@0
    32
#endif 
sl@0
    33
#include <drivers/resourceman.h>
sl@0
    34
sl@0
    35
/** #defines for client bit masks */
sl@0
    36
#define ID_INDEX_BIT_MASK 0x3FFF  /* bit 0 -13 */
sl@0
    37
#define USER_SIDE_CLIENT_BIT_MASK 0x4000 //Bit 14
sl@0
    38
#define CLIENT_THREAD_RELATIVE_BIT_MASK 0x80000000 //Bit 31
sl@0
    39
#define INSTANCE_COUNT_BIT_MASK 0x1FFF //13 Bits
sl@0
    40
#define INSTANCE_COUNT_POS 18
sl@0
    41
#define CLIENT_POWER_CONTROLLER_BIT_MASK 0x8000 //Bit 15
sl@0
    42
/** Bit to indicate valid post boot level in resource */
sl@0
    43
#define SET_VALID_POST_BOOT_LEVEL 0x8000 //Bit 15
sl@0
    44
sl@0
    45
#define RESOURCE_NOT_IN_OPERATION 0x1	
sl@0
    46
#define PRM_DYNAMIC_RESOURCE_INITIAL_SIZE 2
sl@0
    47
sl@0
    48
#define PRM_STATIC_RESOURCE					0x0
sl@0
    49
#define PRM_STATIC_DEPENDENCY_RESOURCE		0x1
sl@0
    50
#define PRM_DYNAMIC_RESOURCE				0x2	
sl@0
    51
#define PRM_DYNAMIC_DEPENDENCY_RESOURCE		0x3
sl@0
    52
#define RESOURCE_BIT_IN_ID_CHECK			16
sl@0
    53
sl@0
    54
sl@0
    55
static const TInt KMaxResourceNameLength=0x20; //Maximum allowable resource length is 32 characters.
sl@0
    56
static const TInt KMaxClientNameLength=0x20; //Maximum allowable client length is 32 characters.
sl@0
    57
sl@0
    58
sl@0
    59
_LIT8(KPowerController, "PowerController");
sl@0
    60
_LIT8(KDfcThread1Name, "DfcThread1");
sl@0
    61
_LIT8(KDfcThread0Name, "DfcThread0");
sl@0
    62
_LIT8(KNullThreadName, "Null");
sl@0
    63
_LIT8(KNoClient, "NoClient");
sl@0
    64
_LIT8(KParentResource, "ParentResource");
sl@0
    65
/** Macro to check the context of client calling RM API.
sl@0
    66
    Panics if it is called from ISR, IDFC, NULL thread or DFC thread1 */
sl@0
    67
#ifdef DEBUG_VERSION
sl@0
    68
#define CHECK_CONTEXT(t)																\
sl@0
    69
	__ASSERT_ALWAYS(NKern::CurrentContext() == NKern::EThread, Panic(ECalledFromIsr));	\
sl@0
    70
	const TDesC8* pDfc1 = &KDfcThread1Name;												\
sl@0
    71
	if(!pDfc1->Compare(*(TDesC8*)t.iName))												\
sl@0
    72
		Panic(ECalledFromDfcThread1);													\
sl@0
    73
    const TDesC8* pNull = &KNullThreadName;												\
sl@0
    74
	if(!pNull->Compare(*(TDesC8*)t.iName))												\
sl@0
    75
		Panic(ECalledFromNullThread);
sl@0
    76
#else
sl@0
    77
#define CHECK_CONTEXT(t)
sl@0
    78
#endif
sl@0
    79
sl@0
    80
/** Macro to unlock and return */
sl@0
    81
#define UNLOCK_RETURN(retval)       \
sl@0
    82
    {                               \
sl@0
    83
    UnLock();                       \
sl@0
    84
    return(retval);                 \
sl@0
    85
    }
sl@0
    86
sl@0
    87
/** Macro to push the item into the specified list. Item are pushed to the head of the list. */
sl@0
    88
#define LIST_PUSH(list,item,link)	\
sl@0
    89
	{                               \
sl@0
    90
	(item)->link = (list);			\
sl@0
    91
	(list) = (item);                \
sl@0
    92
	}
sl@0
    93
sl@0
    94
/** Macro to pop the item from the specified list. Item are poped from the head of the list. */
sl@0
    95
#define LIST_POP(list,item,link)	\
sl@0
    96
	{                               \
sl@0
    97
	(item) = (list);				\
sl@0
    98
	if ((item))						\
sl@0
    99
		{							\
sl@0
   100
		(list) = (item)->link;		\
sl@0
   101
		(item)->link = NULL;		\
sl@0
   102
		}                           \
sl@0
   103
	}
sl@0
   104
sl@0
   105
/** Macro to remove the item from the list. */
sl@0
   106
#define LIST_REMOVE(list,item,link,className)		\
sl@0
   107
	if (list)										\
sl@0
   108
		{											\
sl@0
   109
		className* current = (list);				\
sl@0
   110
		if (current==(item))						\
sl@0
   111
			{										\
sl@0
   112
			(list) = (item)->link;					\
sl@0
   113
			(item)->link = NULL;					\
sl@0
   114
			}										\
sl@0
   115
		else										\
sl@0
   116
			{										\
sl@0
   117
			className* next = current->link;		\
sl@0
   118
			while (next)							\
sl@0
   119
				{									\
sl@0
   120
				if ((item)==next)					\
sl@0
   121
					{								\
sl@0
   122
					current->link=next->link;		\
sl@0
   123
					next->link = NULL;				\
sl@0
   124
					break;							\
sl@0
   125
					}								\
sl@0
   126
				current = next;						\
sl@0
   127
				next = next->link;					\
sl@0
   128
				}									\
sl@0
   129
			}										\
sl@0
   130
		}
sl@0
   131
sl@0
   132
sl@0
   133
/* Macro to add dynamic resource to appropriate containers. Used only in extended version */
sl@0
   134
#define ADD_TO_RESOURCE_CONTAINER(list, res, resId, resIdCount)				\
sl@0
   135
	{																		\
sl@0
   136
	TUint16 growBy = (list).GrowBy();										\
sl@0
   137
	if(!growBy)																\
sl@0
   138
		(list).Initialise((TUint16)PRM_DYNAMIC_RESOURCE_INITIAL_SIZE);		\
sl@0
   139
	if((list).Add(res, resId) == KErrNoMemory)								\
sl@0
   140
		{																	\
sl@0
   141
		TInt r = (list).ReSize(growBy);										\
sl@0
   142
		if(r != KErrNone)													\
sl@0
   143
			return r;														\
sl@0
   144
		(list).Add(res, resId);												\
sl@0
   145
		}																	\
sl@0
   146
	res->iResourceId |= resId;												\
sl@0
   147
	resId = res->iResourceId;												\
sl@0
   148
	resIdCount++;															\
sl@0
   149
	}
sl@0
   150
	
sl@0
   151
/* Macro to get the resource from appropriate list. Used only in extended version */
sl@0
   152
#define GET_RESOURCE_FROM_LIST(resId, res)														\
sl@0
   153
	{																							\
sl@0
   154
	switch((resId >> RESOURCE_BIT_IN_ID_CHECK) & 0x3)											\
sl@0
   155
		{																						\
sl@0
   156
		case PRM_STATIC_RESOURCE:																\
sl@0
   157
			if(resId > iStaticResourceArrayEntries)												\
sl@0
   158
				UNLOCK_RETURN(KErrNotFound);													\
sl@0
   159
			res = iStaticResourceArray[resId - 1];												\
sl@0
   160
			if(!res)																			\
sl@0
   161
				UNLOCK_RETURN(KErrNotFound);													\
sl@0
   162
			break;																				\
sl@0
   163
		case PRM_STATIC_DEPENDENCY_RESOURCE:													\
sl@0
   164
			if((TUint16)(resId & ID_INDEX_BIT_MASK) > iStaticResDependencyCount)				\
sl@0
   165
				UNLOCK_RETURN(KErrNotFound);													\
sl@0
   166
			res = iStaticResDependencyArray[(TUint16)(resId & ID_INDEX_BIT_MASK)  - 1];			\
sl@0
   167
			break;																				\
sl@0
   168
		case PRM_DYNAMIC_RESOURCE:																\
sl@0
   169
			res = iDynamicResourceList[(TUint16)(resId & ID_INDEX_BIT_MASK)];					\
sl@0
   170
			if(!res)																			\
sl@0
   171
				UNLOCK_RETURN(KErrNotFound);													\
sl@0
   172
			break;																				\
sl@0
   173
		case PRM_DYNAMIC_DEPENDENCY_RESOURCE:													\
sl@0
   174
			res = iDynamicResDependencyList[(TUint16)(resId & ID_INDEX_BIT_MASK)];				\
sl@0
   175
			if(!res)																			\
sl@0
   176
				UNLOCK_RETURN(KErrNotFound);													\
sl@0
   177
			break;																				\
sl@0
   178
		default:																				\
sl@0
   179
			UNLOCK_RETURN(KErrArgument);														\
sl@0
   180
		}																						\
sl@0
   181
	}
sl@0
   182
sl@0
   183
/**Macro to get the client from appropriate client list based on bit 14 of client ID.
sl@0
   184
   If the client is registered as thread relative, then check is made to make sure
sl@0
   185
   it is called from the same thread. */
sl@0
   186
#define VALIDATE_CLIENT(t)																						\
sl@0
   187
	if(aClientId & USER_SIDE_CLIENT_BIT_MASK)																	\
sl@0
   188
		pC = iUserSideClientList[(TUint16)(aClientId & ID_INDEX_BIT_MASK)];										\
sl@0
   189
	else																										\
sl@0
   190
		pC = iClientList[(TUint16)(aClientId & ID_INDEX_BIT_MASK)];												\
sl@0
   191
	if(!pC)																										\
sl@0
   192
		{																										\
sl@0
   193
		__KTRACE_OPT(KRESMANAGER, Kern::Printf("Client ID not Found"));											\
sl@0
   194
		UNLOCK_RETURN(KErrAccessDenied);																		\
sl@0
   195
		}																										\
sl@0
   196
	if(pC->iClientId  != aClientId)																				\
sl@0
   197
		{																										\
sl@0
   198
		__KTRACE_OPT(KRESMANAGER, Kern::Printf("Client ID does not match"));									\
sl@0
   199
		UNLOCK_RETURN(KErrAccessDenied);																		\
sl@0
   200
		}																										\
sl@0
   201
	if(pC->iClientId & CLIENT_THREAD_RELATIVE_BIT_MASK)															\
sl@0
   202
		{																										\
sl@0
   203
		if(pC->iThreadId != t.iId)																				\
sl@0
   204
			{																									\
sl@0
   205
			__KTRACE_OPT(KRESMANAGER, Kern::Printf("Client not called from thread context(Thread Relative)"));	\
sl@0
   206
			UNLOCK_RETURN(KErrAccessDenied);																	\
sl@0
   207
			}																									\
sl@0
   208
		}
sl@0
   209
sl@0
   210
/** Macro to get the target client from appropriate client list based on bit 14 of client ID. */
sl@0
   211
#define GET_TARGET_CLIENT()																				\
sl@0
   212
	if(aTargetClientId & USER_SIDE_CLIENT_BIT_MASK) 													\
sl@0
   213
		pC = iUserSideClientList[(TUint16)(aTargetClientId & ID_INDEX_BIT_MASK)];	    				\
sl@0
   214
	else																								\
sl@0
   215
		pC = iClientList[(TUint16)(aTargetClientId & ID_INDEX_BIT_MASK)];								\
sl@0
   216
	if(!pC)																								\
sl@0
   217
		{																								\
sl@0
   218
		__KTRACE_OPT(KRESMANAGER, Kern::Printf("Target Client ID not found"));							\
sl@0
   219
		UNLOCK_RETURN(KErrNotFound);																	\
sl@0
   220
		}																								\
sl@0
   221
	if(pC->iClientId != aTargetClientId)																\
sl@0
   222
		{																								\
sl@0
   223
		__KTRACE_OPT(KRESMANAGER, Kern::Printf("Client ID does not match"));							\
sl@0
   224
		UNLOCK_RETURN(KErrNotFound);																	\
sl@0
   225
		}
sl@0
   226
sl@0
   227
/* Macro definition for entry point of Power Resource Controller */
sl@0
   228
#define DECLARE_RESOURCE_MANAGER_EXTENSION(TheController)															\
sl@0
   229
	TDfc* resourceInitDfc = NULL;																					\
sl@0
   230
	static void ResourceInit(TAny* aController)																		\
sl@0
   231
		{																											\
sl@0
   232
		TInt aReason = NKern::EThread;																				\
sl@0
   233
		PRM_BOOTING_TRACE																							\
sl@0
   234
		((DPowerResourceController*)aController)->InitResources();													\
sl@0
   235
		delete resourceInitDfc;																						\
sl@0
   236
		return;																										\
sl@0
   237
		}																											\
sl@0
   238
	void CreateController();																						\
sl@0
   239
	GLDEF_C TInt KernelModuleEntry(TInt aReason)																	\
sl@0
   240
		{																											\
sl@0
   241
		if(aReason==KModuleEntryReasonVariantInit0)																	\
sl@0
   242
			{																										\
sl@0
   243
			__KTRACE_OPT(KBOOT, Kern::Printf("Create Resource Controller"));										\
sl@0
   244
			CreateController();																						\
sl@0
   245
			return KErrNone;																						\
sl@0
   246
			}																										\
sl@0
   247
		if (aReason==KModuleEntryReasonExtensionInit0)																\
sl@0
   248
			return KExtensionMaximumPriority;																		\
sl@0
   249
		if (aReason!=KModuleEntryReasonExtensionInit1)																\
sl@0
   250
			return KErrArgument;																					\
sl@0
   251
		PRM_BOOTING_TRACE																							\
sl@0
   252
		__KTRACE_OPT(KBOOT, Kern::Printf("Initialise Resource Controller"));										\
sl@0
   253
		TInt r = KErrNone;																							\
sl@0
   254
		r = DPowerResourceController::InitController();																\
sl@0
   255
		if(r != KErrNone)																							\
sl@0
   256
			return r;																								\
sl@0
   257
		__KTRACE_OPT(KBOOT, Kern::Printf("Create PDD and queue ResourceInit DFC"));									\
sl@0
   258
		DResConPddFactory* device = new DResConPddFactory;															\
sl@0
   259
		if(!device)																									\
sl@0
   260
			return KErrNoMemory;																					\
sl@0
   261
		r = Kern::InstallPhysicalDevice(device);																	\
sl@0
   262
		if(r != KErrNone)																							\
sl@0
   263
			return r;																								\
sl@0
   264
		resourceInitDfc = new TDfc(&ResourceInit,(TAny*)&TheController,Kern::SvMsgQue(),KMaxDfcPriority-1);			\
sl@0
   265
		if(!resourceInitDfc)																						\
sl@0
   266
			return KErrNoMemory;																					\
sl@0
   267
		resourceInitDfc->Enque();																					\
sl@0
   268
       	return KErrNone;																							\
sl@0
   269
		}																											\
sl@0
   270
		GLDEF_C void CreateController()
sl@0
   271
sl@0
   272
struct SPowerResourceClient;
sl@0
   273
struct TPowerRequest;
sl@0
   274
struct SPowerRequest;
sl@0
   275
struct SPowerResourceClientLevel;
sl@0
   276
struct SIdleResourceInfo;
sl@0
   277
class DPowerResourceController;
sl@0
   278
sl@0
   279
/**
sl@0
   280
@internalComponent
sl@0
   281
@prototype 9.5
sl@0
   282
Interface class for Resource Manager
sl@0
   283
Functions from PowerResourceManager calls corresponding functions of this 
sl@0
   284
class which in turn calls Powercontroller functions.
sl@0
   285
*/
sl@0
   286
class TInterface
sl@0
   287
	{
sl@0
   288
public:
sl@0
   289
    static TInt RegisterClient(TUint& aClientId, const TDesC8& aName, TOwnerType aType=EOwnerProcess);
sl@0
   290
    static TInt DeRegisterClient(TUint aClientId);
sl@0
   291
    static TInt GetClientName(TUint aClientId, TUint aTargetClientId, TDes8& aName);
sl@0
   292
    static TInt GetClientId(TUint aClientId, TDesC8& aClientName, TUint& aTargetClientId);
sl@0
   293
    static TInt GetResourceId(TUint aClientId, TDesC8& aResourceName, TUint& aResourceId);
sl@0
   294
    static TInt GetResourceInfo(TUint aClientId, TUint aResourceId, TAny* aInfo);
sl@0
   295
    static TInt GetNumResourcesInUseByClient(TUint aClientId, TUint aTargetClientId, TUint& aNumResources);
sl@0
   296
    static TInt GetInfoOnResourcesInUseByClient(TUint aClientId, TUint aTargetClientId, TUint& aNumResources, TAny* aInfo);
sl@0
   297
    static TInt GetNumClientsUsingResource(TUint aClientId, TUint aResourceId, TUint& aNumClients);
sl@0
   298
    static TInt GetInfoOnClientsUsingResource(TUint aClientId, TUint aResourceId, TUint& aNumClients, TAny* aInfo);
sl@0
   299
    static TInt AllocReserve(TUint aClientId, TUint8 aNumCl, TUint8 aNumRm);
sl@0
   300
    static TInt ChangeResourceState(TUint aClientId, TUint aResourceId, TInt aNewState, TPowerResourceCb* aCb=NULL);
sl@0
   301
    static TInt GetResourceState(TUint aClientId, TUint aResourceId, TBool aCached, TInt& aState, TInt& aLevelOwnerId);
sl@0
   302
    static TInt GetResourceState(TUint aClientId, TUint aResourceId, TBool aCached, TPowerResourceCb& aCb);
sl@0
   303
    static TInt CancelAsyncRequestCallBack(TUint aClientId, TUint aResourceId, TPowerResourceCb& aCb);
sl@0
   304
    static TInt RequestNotification(TUint aClientId, TUint aResourceId, DPowerResourceNotification& aN);
sl@0
   305
    static TInt RequestNotification(TUint aClientId, TUint aResourceId, DPowerResourceNotification& aN, TInt aThreshold, TBool aDirection);
sl@0
   306
    static TInt CancelNotification(TUint aClientId, TUint aResourceId, DPowerResourceNotification& aN);
sl@0
   307
	static TInt DeRegisterClientLevelFromResource(TUint aClientId, TUint aResourceId);
sl@0
   308
    static DPowerResourceController* GetPowerResourceController(void);
sl@0
   309
    static TInt ControlIO(TUint aClientId, TUint aFunction, TAny* aParam1, TAny* aParam2, TAny* aParam3);
sl@0
   310
	};
sl@0
   311
sl@0
   312
/**
sl@0
   313
@internalComponent
sl@0
   314
@prototype 9.5
sl@0
   315
Container class to create containers of pointers to clients.
sl@0
   316
*/
sl@0
   317
template <class T>
sl@0
   318
class DResourceCon : public DBase
sl@0
   319
	{
sl@0
   320
public:
sl@0
   321
    inline TInt Initialise(TUint16 aInitialSize);
sl@0
   322
    inline void Delete();
sl@0
   323
    inline T*  operator[](TUint16 aIndex);
sl@0
   324
    inline TInt Remove(T* aObj, TUint16 aIndex);
sl@0
   325
    inline TInt Add(T* aObj, TUint &aId);
sl@0
   326
    inline TInt Find(T*& anEntry, TDesC& aName);
sl@0
   327
    inline TInt ReSize(TUint16 aGrowBy);
sl@0
   328
    inline TUint16 Count() {return iCount;}
sl@0
   329
    inline TUint16 Allocd() {return iAllocated;}
sl@0
   330
	inline TUint16 GrowBy() {return iGrowBy;}
sl@0
   331
private:
sl@0
   332
    TUint16 iGrowBy; //Size to grow the size of the array.
sl@0
   333
    TUint16 iAllocated;  //Size of the array
sl@0
   334
    TUint16 iCount; //Valid entries in the array
sl@0
   335
    TUint16 iInstanceCount; //FreeCounter incremented whenever an entry is added.
sl@0
   336
    TUint16 iFreeLoc; //Cached free location in the array
sl@0
   337
    TUint16 iSpare;
sl@0
   338
    T** iArray;
sl@0
   339
	};
sl@0
   340
sl@0
   341
/** 
sl@0
   342
@internalComponent
sl@0
   343
@prototype 9.5
sl@0
   344
Factory class for physical device 
sl@0
   345
*/
sl@0
   346
NONSHARABLE_CLASS(DResConPddFactory) : public DPhysicalDevice
sl@0
   347
	{
sl@0
   348
public:
sl@0
   349
	/**
sl@0
   350
	Structure for holding PDD capabilities information
sl@0
   351
	*/
sl@0
   352
	class TCaps
sl@0
   353
		{
sl@0
   354
	public:
sl@0
   355
		TVersion iVersion;
sl@0
   356
		};
sl@0
   357
public:
sl@0
   358
    DResConPddFactory();
sl@0
   359
    virtual TInt Install();
sl@0
   360
    virtual TInt Create(DBase*& aChannel, TInt aUnit, const TDesC8* anInfo, const TVersion& aVer);
sl@0
   361
    virtual TInt Validate(TInt aUint, const TDesC8* anInfo, const TVersion& aVer);
sl@0
   362
    virtual void GetCaps(TDes8& aDes) const;
sl@0
   363
    inline static TVersion VersionRequired();
sl@0
   364
	};
sl@0
   365
sl@0
   366
/**
sl@0
   367
@internalComponent
sl@0
   368
@prototype 9.5
sl@0
   369
Interface class for user side resource controller proxy. For each user side channel opened an object of 
sl@0
   370
this class is created in heap and pointer to resource controller is stored in iController member variable. 
sl@0
   371
User side resource controller proxy calls the resource controller API's by deferencing the pointer. 
sl@0
   372
This class is required as when the channel is closed the device driver framework tries to delete 
sl@0
   373
the object stored in ipdd, because of which it is not possible to pass the controller pointer directly.
sl@0
   374
*/
sl@0
   375
class DUserSideProxyInterface: public DBase
sl@0
   376
	{
sl@0
   377
	public:
sl@0
   378
	DPowerResourceController *iController;
sl@0
   379
	};
sl@0
   380
sl@0
   381
sl@0
   382
/** 
sl@0
   383
@publishedPartner
sl@0
   384
@prototype 9.5
sl@0
   385
resource manager implementation base class
sl@0
   386
*/
sl@0
   387
NONSHARABLE_CLASS (DPowerResourceController) : public DBase
sl@0
   388
	{
sl@0
   389
public:
sl@0
   390
    TInt RegisterClient(TUint& aClientId, const TDesC8& aName, TOwnerType aType=EOwnerProcess);
sl@0
   391
    TInt DeRegisterClient(TUint aClientId);
sl@0
   392
    virtual TInt GetClientName(TUint aClientId, TUint aTargetClientId, TDes8& aName);
sl@0
   393
    virtual TInt GetClientId(TUint aClientId, TDesC8& aClientName, TUint& aTargetClientId);
sl@0
   394
    virtual TInt GetResourceId(TUint aClientId, TDesC8& aResourceName, TUint& aResourceId);
sl@0
   395
    virtual TInt GetResourceInfo(TUint aClientId, TUint aResourceId, TAny* aInfo);
sl@0
   396
    virtual TInt GetNumResourcesInUseByClient(TUint aClientId, TUint aTargetClientId, TUint& aNumResources);
sl@0
   397
    virtual TInt GetInfoOnResourcesInUseByClient(TUint aClientId, TUint aTargetClientId, TUint& aNumResources, TAny* aInfo);
sl@0
   398
    virtual TInt GetNumClientsUsingResource(TUint aClientId, TUint aResourceId, TUint& aNumClients);
sl@0
   399
    virtual TInt GetInfoOnClientsUsingResource(TUint aClientId, TUint aResourceId, TUint& aNumClients, TAny* aInfo);
sl@0
   400
    virtual TInt AllocReserve(TUint aClientId, TUint8 aNumCl, TUint8 aNumRm);
sl@0
   401
    virtual TInt ChangeResourceState(TUint aClientId, TUint aResourceId, TInt aNewState, TPowerResourceCb* aCb=NULL);
sl@0
   402
    virtual TInt GetResourceState(TUint aClientId, TUint aResourceId, TBool aCached, TInt& aState, TInt& aLevelOwnerId);
sl@0
   403
    virtual TInt GetResourceState(TUint aClientId, TUint aResourceId, TBool aCached, TPowerResourceCb& aCb);
sl@0
   404
    virtual TInt CancelAsyncRequestCallBack(TUint aClientId, TUint aResourceId, TPowerResourceCb& aCb);
sl@0
   405
    virtual TInt RequestNotification(TUint aClientId, TUint aResourceId, DPowerResourceNotification& aN);
sl@0
   406
    virtual TInt RequestNotification(TUint aClientId, TUint aResourceId, DPowerResourceNotification& aN, TInt aThreshold, TBool aDirection);
sl@0
   407
    virtual TInt CancelNotification(TUint aClientId, TUint aResourceId, DPowerResourceNotification& aN);
sl@0
   408
    virtual TInt DeRegisterClientLevelFromResource(TUint aClientId, TUint aResourceId);
sl@0
   409
public:
sl@0
   410
	enum TResConPanic
sl@0
   411
		{
sl@0
   412
		ECalledFromDfcThread0 = 0,
sl@0
   413
		ECalledFromIsr = 1,
sl@0
   414
		ECalledFromNullThread = 2,
sl@0
   415
		ECalledFromDfcThread1 = 3,
sl@0
   416
		EClientHasPendingAsyncRequest = 4,
sl@0
   417
		EClientHasNotificationObject = 5,
sl@0
   418
		EControllerAlreadyExists = 6,
sl@0
   419
		ECustomFunctionNotSet = 7,
sl@0
   420
		EClientIdNotInClientLevelList = 8,
sl@0
   421
		ENoMemToCreatePowerControllerClient = 9,
sl@0
   422
		EResourceNameExceedsLimit = 10, 
sl@0
   423
		EObjectNotFoundInList = 11 
sl@0
   424
		};
sl@0
   425
#ifdef PRM_ENABLE_EXTENDED_VERSION	
sl@0
   426
	enum TExtendedResConPanic
sl@0
   427
		{
sl@0
   428
		EClosedLoopDependencies = EObjectNotFoundInList + 2, //13
sl@0
   429
		ERegisteringNonDependentStaticResource = 14,
sl@0
   430
		EClientHasDynamicResourceRegistered = 15,
sl@0
   431
		EDynamicResourceStillRegistered = 16,
sl@0
   432
		ERegisteringDependentStaticResourceWithHoles = 17
sl@0
   433
		};
sl@0
   434
#endif
sl@0
   435
	enum TResConStartSequence
sl@0
   436
		{
sl@0
   437
		EResConCreated,
sl@0
   438
		EResConInitialised,
sl@0
   439
		EResConStartupCompleted
sl@0
   440
		};
sl@0
   441
    //early initialization
sl@0
   442
    IMPORT_C static TInt InitController();
sl@0
   443
    TInt InitResources();
sl@0
   444
    //request a post-boot level for the resource
sl@0
   445
    IMPORT_C static TInt PostBootLevel(TUint aResId, TInt aLevel);
sl@0
   446
    //request registration of static resource
sl@0
   447
    IMPORT_C static TInt RegisterStaticResource(TUint aClientId, DStaticPowerResource* pR);
sl@0
   448
	//request registration of group/array of static resources
sl@0
   449
	IMPORT_C static TInt RegisterArrayOfStaticResources(TUint aClientId, DStaticPowerResource**& aStaticResourceArray, TUint aResCount);
sl@0
   450
    //registration for proxy client
sl@0
   451
    virtual TInt RegisterProxyClient(TUint& aProxyId, const TDesC8& aName);
sl@0
   452
    virtual TInt DeregisterProxyClient(TUint aClientId);
sl@0
   453
    //register list of resources whose state matter to Idle
sl@0
   454
    virtual TInt RegisterResourcesForIdle(TInt aPowerControllerId, TUint aNumResources, TPtr* aBuf);
sl@0
   455
    static void Panic(TUint8 aPanic);
sl@0
   456
    virtual TInt GetInterface(TUint aClientId, TUint aInterfaceId, TAny* aParam1, TAny* aParam2, TAny* aParam3);
sl@0
   457
	virtual ~DPowerResourceController();
sl@0
   458
	/**@internalComponent*/
sl@0
   459
	void CompleteNotifications(TInt aClientId, DStaticPowerResource* aResource, TInt aState, TInt aReturnCode, TInt aLevelOwnerId, TBool aLock = ETrue);
sl@0
   460
#ifdef PRM_ENABLE_EXTENDED_VERSION
sl@0
   461
	/**@internalComponent*/
sl@0
   462
	TInt ReserveClientLevelPoolCount(TUint16 aCount);
sl@0
   463
	/**@internalComponent*/
sl@0
   464
	void RemoveClientLevelFromPool(SPowerResourceClientLevel *&aLevelPtr);
sl@0
   465
	/**@internalComponent*/
sl@0
   466
	TInt HandleResourceChange(TPowerRequest &aRequest, TPropagation aProp, TUint aOriginatorId,
sl@0
   467
									const TDesC8& aOriginatorName, DStaticPowerResourceD* aResource);
sl@0
   468
#endif
sl@0
   469
protected:
sl@0
   470
    //generic layer function to be called by the PSL
sl@0
   471
    DPowerResourceController();
sl@0
   472
    void SetDfcQ(TDfcQue* aDfcQ);
sl@0
   473
	#ifdef PRM_ENABLE_EXTENDED_VERSION
sl@0
   474
    void SetDfcQDependency(TDfcQue* aDfcQ);
sl@0
   475
	#endif
sl@0
   476
    TInt InitPools(TUint16 aKClients, TUint16 aUClients, TUint16 aNClientLevels, TUint16 aNRequests);
sl@0
   477
	/* Lock the resource controller mutex */
sl@0
   478
	inline void Lock()	{ NKern::ThreadEnterCS();
sl@0
   479
						  Kern::MutexWait(*iResourceMutex); }
sl@0
   480
	inline void UnLock()	{ Kern::MutexSignal(*iResourceMutex);
sl@0
   481
							  NKern::ThreadLeaveCS();}
sl@0
   482
#ifdef PRM_ENABLE_EXTENDED_VERSION
sl@0
   483
	//Default implementation, PSL re-implements these if features supported
sl@0
   484
	virtual TInt DoRegisterStaticResourcesDependency(DStaticPowerResourceD**& aStaticResourceDArray, TUint16& aStaticResourceDCount);
sl@0
   485
#endif
sl@0
   486
private:
sl@0
   487
    // pure virtual implemented by PSL - to be called by PIL
sl@0
   488
    virtual TInt DoInitController()=0;
sl@0
   489
    virtual TInt DoRegisterStaticResources(DStaticPowerResource**& aStaticResourceArray, TUint16& aStaticResourceCount)=0;
sl@0
   490
    /**@internalComponent*/
sl@0
   491
    TInt CheckLevelAndAddClient(SPowerResourceClient* pC, TPowerRequest* Request);
sl@0
   492
    static void MsgQFunc(TAny* aPtr);
sl@0
   493
    #ifdef PRM_ENABLE_EXTENDED_VERSION
sl@0
   494
    static void MsgQDependencyFunc(TAny* aPtr);
sl@0
   495
    #endif
sl@0
   496
    
sl@0
   497
	/**@internalComponent*/
sl@0
   498
	void ResourceStateChangeOfClientLevels(SPowerResourceClient* pC);
sl@0
   499
	/**@internalComponent*/
sl@0
   500
    void HandleMsg(TPowerRequest& aRequest);
sl@0
   501
	#ifdef PRM_ENABLE_EXTENDED_VERSION
sl@0
   502
    /**@internalComponent*/
sl@0
   503
    void HandleDependencyMsg(TPowerRequest& aRequest);
sl@0
   504
	#endif
sl@0
   505
	/**@internalComponent*/
sl@0
   506
    void CompleteRequest(TPowerRequest& aRequest);
sl@0
   507
	/**@internalComponent*/
sl@0
   508
    void MoveRequestToFreePool(TPowerRequest *aReq);
sl@0
   509
	/**@internalComponent*/
sl@0
   510
	TInt HandleReservationOfObjects(TPowerRequest& aRequest);
sl@0
   511
	/**@internalComponent*/
sl@0
   512
	TInt HandleClientRegistration(TPowerRequest& aRequest);
sl@0
   513
#ifdef PRM_ENABLE_EXTENDED_VERSION
sl@0
   514
	TInt RegisterDynamicResource(SPowerResourceClient* aClientPtr, DDynamicPowerResource* aPDRes, TUint* aDynamicResourceId);
sl@0
   515
	TInt DeregisterDynamicResource(SPowerResourceClient* aClientPtr, TUint aDynamicResourceId, TInt* aPDefLevel);
sl@0
   516
	TInt RegisterResourceDependency(SPowerResourceClient* aClientPtr, SResourceDependencyInfo* aInfo1, SResourceDependencyInfo* aInfo2);
sl@0
   517
	/**@internalComponent*/
sl@0
   518
	void CheckForDependencyLoop(DStaticPowerResourceD* pR, TUint aParentResId, TUint aTargetResId);
sl@0
   519
	TInt DeregisterResourceDependency(SPowerResourceClient* aClientPtr, TUint aResId1, TUint aResId2);
sl@0
   520
	/**@internalComponent*/
sl@0
   521
	TInt HandleDependencyResourceStateChange(SPowerResourceClient* pC, TPowerRequest& aRequest);
sl@0
   522
	TInt GetNumDependentsForResource(TUint aResourceId, TUint* aNumResources);
sl@0
   523
	TInt GetDependentsIdForResource(TUint aResourceId, TAny* aInfo, TUint* aNumDepResources);
sl@0
   524
	TInt HandleResourceRegistration(TPowerRequest& aReq);
sl@0
   525
#endif
sl@0
   526
public:
sl@0
   527
	DMutex* iResourceMutex;
sl@0
   528
protected:
sl@0
   529
    TDfcQue* iDfcQ;
sl@0
   530
    TMessageQue *iMsgQ;
sl@0
   531
#ifdef PRM_ENABLE_EXTENDED_VERSION
sl@0
   532
	TDfcQue* iDfcQDependency;
sl@0
   533
	TMessageQue* iMsgQDependency;
sl@0
   534
	TBool iDfcQDependencyLock;
sl@0
   535
#endif
sl@0
   536
private:
sl@0
   537
    DStaticPowerResource** iStaticResourceArray;
sl@0
   538
    DResourceCon<SPowerResourceClient> iClientList;
sl@0
   539
    DResourceCon<SPowerResourceClient> iUserSideClientList;
sl@0
   540
#ifdef RESOURCE_MANAGER_SIMULATED_PSL
sl@0
   541
	RPointerArray<SPowerResourceClient> iCleanList;
sl@0
   542
#endif
sl@0
   543
    SPowerResourceClient* iClientPool;
sl@0
   544
    SPowerRequest* iRequestPool;
sl@0
   545
    SPowerResourceClientLevel* iClientLevelPool;
sl@0
   546
	TUint iPowerControllerId; //Stores the ID allocated to PowerController
sl@0
   547
    SIdleResourceInfo* iListForIdle;
sl@0
   548
    TUint iInitialised;
sl@0
   549
    TUint16 iClientCount;
sl@0
   550
    TUint16 iUserSideClientCount;
sl@0
   551
    TUint16 iClientLevelPoolCount;
sl@0
   552
    TUint16 iClientLevelPoolGrowBy;
sl@0
   553
    TUint16 iRequestPoolCount;
sl@0
   554
    TUint16 iRequestPoolGrowBy;
sl@0
   555
    TUint16 iStaticResourceArrayEntries; //Number of entries in the array including holes if any.
sl@0
   556
	TUint16 iStaticResourceCount;  //Actual number of static resources registered (valid entries).
sl@0
   557
	TUint	iReserved2; //Reserved for future use
sl@0
   558
#ifdef PRM_ENABLE_EXTENDED_VERSION
sl@0
   559
	DResourceCon<DDynamicPowerResource> iDynamicResourceList;
sl@0
   560
	DResourceCon<DDynamicPowerResourceD> iDynamicResDependencyList;
sl@0
   561
	DStaticPowerResourceD** iStaticResDependencyArray;
sl@0
   562
	SPowerResourceClientLevel* iResourceLevelPool;
sl@0
   563
	TUint16 iResourceLevelPoolCount;
sl@0
   564
	TUint16 iStaticResDependencyCount;
sl@0
   565
	TUint16 iDynamicResourceCount;
sl@0
   566
	TUint8 iDynamicResDependencyCount;
sl@0
   567
	TUint8 iSpare2;
sl@0
   568
	TUint  iReserved3; //Reserved for future use.
sl@0
   569
#endif
sl@0
   570
	};
sl@0
   571
sl@0
   572
/**
sl@0
   573
@publishedPartner
sl@0
   574
@prototype 9.5
sl@0
   575
power level of client in a shared resource
sl@0
   576
*/
sl@0
   577
struct SPowerResourceClientLevel : public SDblQueLink
sl@0
   578
	{
sl@0
   579
    TUint iClientId;
sl@0
   580
    TUint iResourceId;
sl@0
   581
    TInt iLevel;
sl@0
   582
    SPowerResourceClientLevel* iNextInList;
sl@0
   583
	};
sl@0
   584
sl@0
   585
/**
sl@0
   586
@internalComponent 
sl@0
   587
@prototype 9.5
sl@0
   588
respresent client in resource manager
sl@0
   589
*/
sl@0
   590
struct SPowerResourceClient
sl@0
   591
	{
sl@0
   592
    TUint iClientId;
sl@0
   593
    const TDesC8* iName;
sl@0
   594
    SPowerResourceClient* iNextInList;
sl@0
   595
    SPowerResourceClientLevel* iLevelList;
sl@0
   596
    DPowerResourceNotification* iNotificationList;
sl@0
   597
    TUint8 iReservedCl;
sl@0
   598
    TUint8 iReservedRm;
sl@0
   599
    TUint8 iPendingReqCount;
sl@0
   600
    TUint8 iUnderFlowRmCount;
sl@0
   601
    TUint8 iUnderFlowClCount;
sl@0
   602
	TUint8 iDynamicResCount; //Counter for dynamic resource registered by the client. Used only in extended version
sl@0
   603
	TUint8 iSpare1;
sl@0
   604
	TUint8 iSpare2;
sl@0
   605
    union
sl@0
   606
       {
sl@0
   607
       TUint iThreadId;
sl@0
   608
       TAny* iSpare3;
sl@0
   609
       };
sl@0
   610
	};
sl@0
   611
sl@0
   612
/**
sl@0
   613
@publishedPartner
sl@0
   614
@prototype 9.5
sl@0
   615
represents a request inside the resource manager
sl@0
   616
*/
sl@0
   617
struct TPowerRequest : public TThreadMessage
sl@0
   618
	{
sl@0
   619
    /** requests can either be to get the resource value or to change the resource value*/
sl@0
   620
    enum TReqType {EGet, EChange, ESetDefaultLevel, ERegisterKernelClient, ERegisterUsersideClient, EAllocReserve,
sl@0
   621
					ERegisterDynamicResource	};
sl@0
   622
    /** @return thread's own message and turn into a power request. Used for sync/instant calls*/
sl@0
   623
    inline static TPowerRequest& Get()
sl@0
   624
            {return (TPowerRequest&)Kern::Message();}
sl@0
   625
    /** @return type of request get or set */
sl@0
   626
    inline TReqType& ReqType()		// one of TReqType
sl@0
   627
            {return *(TReqType*)&iValue;}
sl@0
   628
    /** @return resource id which is being requested*/
sl@0
   629
    inline TUint& ResourceId()
sl@0
   630
            {return *(TUint*)&iArg[0];}
sl@0
   631
    /** @return id of client making request (only valid on change requests)*/
sl@0
   632
    inline TInt& ClientId()
sl@0
   633
            {return *(TInt*)&iArg[1];}
sl@0
   634
	/**
sl@0
   635
	    On resource state change operations the PIL sets this field with the required level before 
sl@0
   636
		invoking the DoRequest(..) function; on return from DoRequest(..) function the PSL sets this field
sl@0
   637
		with the real state of the resource to be cached by the PIL.On resource state read operations PSL
sl@0
   638
		sets it with the level read.
sl@0
   639
	*/
sl@0
   640
	inline TInt& Level()		
sl@0
   641
            {return *(TInt*)&iArg[2];}
sl@0
   642
    /** @return pointer the resource being requested */
sl@0
   643
    inline DStaticPowerResource*& Resource()
sl@0
   644
            {return *(DStaticPowerResource**)&iArg[3];}
sl@0
   645
    /** @return pointer to resource callback structure, used for async requests */
sl@0
   646
    inline TPowerResourceCb*& ResourceCb()
sl@0
   647
            {return *(TPowerResourceCb**)&iArg[4];}
sl@0
   648
    /** @return return code of resource's DoRequest function when request has been processed */
sl@0
   649
    inline TInt& ReturnCode()
sl@0
   650
            {return *(TInt*)&iArg[5];}
sl@0
   651
    /** @return return ETrue if a change is required on a shared resource */
sl@0
   652
    inline TBool& RequiresChange()
sl@0
   653
            {return *(TInt*)&iArg[6];}
sl@0
   654
	/** @return number of client level objects requested by a client to reserve */
sl@0
   655
	inline TInt& ClientLevelCount()
sl@0
   656
			{return *(TInt*)&iArg[7];}
sl@0
   657
	/** @return number of request objects requested by a client to reserve */
sl@0
   658
	inline TInt& RequestCount()
sl@0
   659
			{return *(TInt*)&iArg[8];}
sl@0
   660
	};
sl@0
   661
sl@0
   662
/**
sl@0
   663
@internalComponent
sl@0
   664
@prototype 9.5
sl@0
   665
*/
sl@0
   666
struct SPowerRequest
sl@0
   667
	{
sl@0
   668
    TPowerRequest iRequest;
sl@0
   669
    SPowerRequest* iNext;
sl@0
   670
	};
sl@0
   671
sl@0
   672
/**
sl@0
   673
@publishedPartner
sl@0
   674
@prototype 9.5
sl@0
   675
Structure representing resource information used for Idle power management
sl@0
   676
*/
sl@0
   677
struct SIdleResourceInfo
sl@0
   678
	{
sl@0
   679
    TUint iResourceId; 
sl@0
   680
    TInt iLevelOwnerId; //Owner of the resource.  
sl@0
   681
    TInt iCurrentLevel; //Cached resource state
sl@0
   682
	TInt iReserved1;	//Reserved for future use.
sl@0
   683
	TInt iReserved2;	//Reserved for future use.
sl@0
   684
	TInt iReserved3;	//Reserved for future use.
sl@0
   685
	};
sl@0
   686
sl@0
   687
#include <drivers/resourcecontrol.inl>
sl@0
   688
sl@0
   689
#endif //__RESOURCECONTROL_H__
sl@0
   690
sl@0
   691