os/graphics/egl/eglrefimpl/src/driver.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) 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
// Reference EGL implementation to support EGL sync objects and OpenWF extensions
sl@0
    15
sl@0
    16
#include "eglprivate.h"
sl@0
    17
#include <EGL/egluids.hrh>
sl@0
    18
#include <e32uid.h>
sl@0
    19
#include <WF/openwfcuids.hrh>
sl@0
    20
sl@0
    21
// OpenWF Composition library
sl@0
    22
//
sl@0
    23
_LIT(KOwfDllName, "libWFC.dll");
sl@0
    24
const TUid KOwfDllUid1 = {KDynamicLibraryUidValue};
sl@0
    25
const TUid KOwfDllUid2 = {KUidSharedDllUidValue};
sl@0
    26
const TUid KOwfDllUid3 = {KUidOpenWfcDllUidValue};
sl@0
    27
const TUidType KOwfDllUidType = TUidType(KOwfDllUid1, KOwfDllUid2, KOwfDllUid3);
sl@0
    28
sl@0
    29
// EGL_KHR_reusable_sync extensions API
sl@0
    30
// Declare here so their addresses can be used to build extensions table
sl@0
    31
//
sl@0
    32
EGLSyncKHR eglCreateSyncKHR(EGLDisplay aDisplay, EGLenum aType, const EGLint *aAttribList);
sl@0
    33
EGLBoolean eglDestroySyncKHR(EGLDisplay aDisplay, EGLSyncKHR aSync);
sl@0
    34
EGLint eglClientWaitSyncKHR(EGLDisplay aDisplay, EGLSyncKHR aSync, EGLint aFlags, EGLTimeKHR aTimeout);
sl@0
    35
EGLBoolean eglSignalSyncKHR(EGLDisplay aDisplay, EGLSyncKHR aSync, EGLenum aMode);
sl@0
    36
EGLBoolean eglGetSyncAttribKHR(EGLDisplay aDisplay, EGLSyncKHR aSync, EGLint aAttribute, EGLint *aValue);
sl@0
    37
sl@0
    38
// Private extensions
sl@0
    39
//
sl@0
    40
EGLint egl_Private_SignalSyncNOK(EGLDisplay aDisplay, EGLSyncKHR aSync, EGLenum aMode);
sl@0
    41
sl@0
    42
// Extensions for shared heap OOM test
sl@0
    43
#ifdef _DEBUG
sl@0
    44
void egliDebugHeapMarkStart();
sl@0
    45
EGLint egliDebugHeapMarkEnd(EGLint aCount);
sl@0
    46
void egliDebugSetBurstAllocFail(EGLenum aType, EGLint aRate, EGLint aBurst);
sl@0
    47
#endif //_DEBUG
sl@0
    48
sl@0
    49
struct TEglNameProcRecord
sl@0
    50
    {
sl@0
    51
    char* iName;
sl@0
    52
    TFuncPtrEglProc iProc;
sl@0
    53
    };
sl@0
    54
sl@0
    55
#define MAKE_NAME_PROC_RECORD(f)    {#f,reinterpret_cast<TFuncPtrEglProc>(f)}
sl@0
    56
sl@0
    57
const TEglNameProcRecord KEglExtensionFuncs[] = {
sl@0
    58
    MAKE_NAME_PROC_RECORD(eglCreateSyncKHR),
sl@0
    59
    MAKE_NAME_PROC_RECORD(eglDestroySyncKHR),
sl@0
    60
    MAKE_NAME_PROC_RECORD(eglClientWaitSyncKHR),
sl@0
    61
    MAKE_NAME_PROC_RECORD(eglSignalSyncKHR),
sl@0
    62
    MAKE_NAME_PROC_RECORD(eglGetSyncAttribKHR),
sl@0
    63
    MAKE_NAME_PROC_RECORD(egl_Private_SignalSyncNOK) 
sl@0
    64
#ifdef _DEBUG
sl@0
    65
    ,
sl@0
    66
    MAKE_NAME_PROC_RECORD(egliDebugHeapMarkStart),
sl@0
    67
    MAKE_NAME_PROC_RECORD(egliDebugHeapMarkEnd),
sl@0
    68
    MAKE_NAME_PROC_RECORD(egliDebugSetBurstAllocFail)
sl@0
    69
#endif //_DEBUG
sl@0
    70
};
sl@0
    71
sl@0
    72
// Use common EglPls() function to hide difference between WINS and target in dealing with PLS
sl@0
    73
//
sl@0
    74
#ifdef __WINS__
sl@0
    75
#include <pls.h>
sl@0
    76
static inline XEglPls* EglPls()
sl@0
    77
	{
sl@0
    78
	const TUid KUidEglDll = {KUidEGLDllUidValue};
sl@0
    79
	return Pls<XEglPls>(KUidEglDll);
sl@0
    80
	}
sl@0
    81
#else
sl@0
    82
static XEglPls TheEglPls;
sl@0
    83
static inline XEglPls* EglPls()
sl@0
    84
	{
sl@0
    85
	return &TheEglPls;
sl@0
    86
	}
sl@0
    87
#endif
sl@0
    88
sl@0
    89
sl@0
    90
// Macros for placement new and delete of shared objects
sl@0
    91
//
sl@0
    92
#define EGL_PLACEMENT_NEW(heap, obj, klass)         klass* obj = NULL; \
sl@0
    93
                                                    TAny* buf = (heap).AllocZ(sizeof(klass)); \
sl@0
    94
                                                    if (buf) \
sl@0
    95
                                                        { \
sl@0
    96
                                                        obj = new(buf) klass((heap)); \
sl@0
    97
                                                        }
sl@0
    98
sl@0
    99
#define EGL_PLACEMENT_DELETE(heap, obj, klass)      obj->~klass(); \
sl@0
   100
                                                    (heap).Free(obj);
sl@0
   101
sl@0
   102
///////////////////////////////////////////////////////////////////////////////
sl@0
   103
// PLS
sl@0
   104
///////////////////////////////////////////////////////////////////////////////
sl@0
   105
sl@0
   106
XEglPls::XEglPls()
sl@0
   107
	{
sl@0
   108
	iError = iLock.CreateLocal();
sl@0
   109
	iDriver = NULL;
sl@0
   110
	}
sl@0
   111
sl@0
   112
XEglPls::~XEglPls()
sl@0
   113
	{
sl@0
   114
	iLock.Close();
sl@0
   115
	}
sl@0
   116
sl@0
   117
///////////////////////////////////////////////////////////////////////////////
sl@0
   118
// EGL driver
sl@0
   119
///////////////////////////////////////////////////////////////////////////////
sl@0
   120
sl@0
   121
CEglDriver::CEglDriver(RHeap& aHeap):
sl@0
   122
	iHeap(aHeap)
sl@0
   123
	{
sl@0
   124
	}
sl@0
   125
sl@0
   126
CEglDriver::~CEglDriver()
sl@0
   127
	{
sl@0
   128
	// EGL display was allocated on shared heap using placement new
sl@0
   129
	if (iDisplay)
sl@0
   130
		{
sl@0
   131
		EGL_PLACEMENT_DELETE(iHeap, iDisplay, CEglDisplay)
sl@0
   132
		}
sl@0
   133
	
sl@0
   134
	iOwfLib.Close();
sl@0
   135
	iLock.Close();
sl@0
   136
	}
sl@0
   137
sl@0
   138
TInt CEglDriver::Open()
sl@0
   139
	{
sl@0
   140
	// we're in trouble if mutex creation failed during DLL initialisation
sl@0
   141
	XEglPls& pls = *EglPls();
sl@0
   142
	if (pls.iError != KErrNone)
sl@0
   143
		{
sl@0
   144
		return pls.iError;
sl@0
   145
		}
sl@0
   146
	
sl@0
   147
	pls.Lock();
sl@0
   148
sl@0
   149
	if (pls.iDriver)
sl@0
   150
		{
sl@0
   151
		++(pls.iDriver->iRefCount);
sl@0
   152
		pls.Unlock();
sl@0
   153
		return KErrNone;
sl@0
   154
		}
sl@0
   155
	
sl@0
   156
	// create shared heap in a local chunk to allow access to shared objects from any threads within Driver
sl@0
   157
	// allow heap to grow from 256 bytes up to 1MB
sl@0
   158
	const TInt KMaxHeapSize = 0x100000;
sl@0
   159
	const TInt KWordAlign = 4;
sl@0
   160
	RHeap* heap = User::ChunkHeap(NULL, KMinHeapSize, KMaxHeapSize, KMinHeapGrowBy, KWordAlign);
sl@0
   161
	if (!heap)
sl@0
   162
		{
sl@0
   163
		pls.Unlock();
sl@0
   164
		return KErrNoMemory;
sl@0
   165
		}
sl@0
   166
	
sl@0
   167
	EGL_PLACEMENT_NEW(*heap, drv, CEglDriver)
sl@0
   168
	if (!drv)
sl@0
   169
		{
sl@0
   170
		heap->Close();
sl@0
   171
		pls.Unlock();
sl@0
   172
		return KErrNoMemory;
sl@0
   173
		}
sl@0
   174
	
sl@0
   175
	const TInt err = drv->Construct();
sl@0
   176
	if (err != KErrNone)
sl@0
   177
		{
sl@0
   178
		EGL_PLACEMENT_DELETE(*heap, drv, CEglDriver)
sl@0
   179
		heap->Close();
sl@0
   180
		pls.Unlock();
sl@0
   181
		
sl@0
   182
		return err;
sl@0
   183
		}
sl@0
   184
	
sl@0
   185
	pls.iDriver = drv;
sl@0
   186
	pls.iDriver->iRefCount = 1;
sl@0
   187
sl@0
   188
	pls.Unlock();
sl@0
   189
	
sl@0
   190
	return KErrNone;
sl@0
   191
	}
sl@0
   192
sl@0
   193
void CEglDriver::Close()
sl@0
   194
	{
sl@0
   195
	XEglPls& pls = *EglPls();
sl@0
   196
	// must never happen if CEglDriver::Open() succeed
sl@0
   197
	__ASSERT_DEBUG(pls.iError == KErrNone, User::Panic(KEglPanicCategory, EEglPanicPlsMutexError));
sl@0
   198
sl@0
   199
	pls.Lock();
sl@0
   200
sl@0
   201
	if (pls.iDriver)
sl@0
   202
		{
sl@0
   203
		if (--(pls.iDriver->iRefCount) == 0)
sl@0
   204
			{
sl@0
   205
			// copy shared heap pointer out as we're about to destroy the driver
sl@0
   206
			RHeap& heap = pls.iDriver->iHeap;
sl@0
   207
			
sl@0
   208
			// driver was allocated on shared heap using placement new
sl@0
   209
			EGL_PLACEMENT_DELETE(heap, pls.iDriver, CEglDriver)
sl@0
   210
			pls.iDriver = NULL;
sl@0
   211
			heap.Close();
sl@0
   212
			}
sl@0
   213
		}
sl@0
   214
sl@0
   215
	pls.Unlock();
sl@0
   216
	}
sl@0
   217
sl@0
   218
CEglDriver* CEglDriver::GetDriver()
sl@0
   219
    {
sl@0
   220
    XEglPls& pls = *EglPls();
sl@0
   221
    return pls.iDriver;    
sl@0
   222
    }
sl@0
   223
sl@0
   224
// called when lock is held
sl@0
   225
//
sl@0
   226
TInt CEglDriver::Construct()
sl@0
   227
	{
sl@0
   228
	TInt err = iLock.CreateLocal();
sl@0
   229
	if (err != KErrNone)
sl@0
   230
		{
sl@0
   231
		return err;
sl@0
   232
		}
sl@0
   233
sl@0
   234
	// create default EGL display as part of driver initialisation, it will be destroyed when the driver is destroyed
sl@0
   235
	// and recreated when driver is reinitialised
sl@0
   236
    EGL_PLACEMENT_NEW(iHeap, disp, CEglDisplay)
sl@0
   237
    if (!disp)
sl@0
   238
        {
sl@0
   239
        return KErrNoMemory;
sl@0
   240
        }
sl@0
   241
sl@0
   242
    iDisplay = disp;
sl@0
   243
	
sl@0
   244
	// for security reason, use TUidType when loading OWF library to make sure it is not a spoofed DLL with the same name
sl@0
   245
	err = iOwfLib.Load(KOwfDllName, KOwfDllUidType);
sl@0
   246
	// allow initialisation to proceed even if OWF library is not found
sl@0
   247
	if (err == KErrNone)
sl@0
   248
	    {
sl@0
   249
        // promote library handle from thread-relative to process-wide to allow closing from any thread
sl@0
   250
        RLibrary dup = iOwfLib;
sl@0
   251
        err = iOwfLib.Duplicate(RThread());
sl@0
   252
        if (err != KErrNone)
sl@0
   253
            {
sl@0
   254
            return err;
sl@0
   255
            }
sl@0
   256
        // if Duplicate succeed, it will overwrite iOwfLib with process-wide handle
sl@0
   257
        // we need to close the original thread-relative handle
sl@0
   258
        dup.Close();
sl@0
   259
    
sl@0
   260
        // In SymbianOS implementation of Khronos API, ordinal #1 is always reserved for internal implementation use as defined in
sl@0
   261
        // SymbianOS standard DEF files for that API. This DEF file ensures BC among different implementations of that API.
sl@0
   262
        // In OWF case, ordinal #1 refers to GetExtensionFunctionTable, a function which will return a table of function pointer.
sl@0
   263
        // One of them is a function that implement GetProcAddress to retrieve OWF extension functions.
sl@0
   264
        //
sl@0
   265
        const TInt KOwfGetFuncTableOrdinalNum = 1;
sl@0
   266
        typedef void (*TFuncPtrOwfGetFuncTable)(TOwfFuncTable*);
sl@0
   267
        
sl@0
   268
        TFuncPtrOwfGetFuncTable owfGetFuncTable = reinterpret_cast<TFuncPtrOwfGetFuncTable>(iOwfLib.Lookup(KOwfGetFuncTableOrdinalNum));
sl@0
   269
        if (!owfGetFuncTable)
sl@0
   270
            {
sl@0
   271
            return KErrNotFound;
sl@0
   272
            }
sl@0
   273
    
sl@0
   274
        owfGetFuncTable(&iOwfFuncTable);
sl@0
   275
    
sl@0
   276
        if (!iOwfFuncTable.iOwfGetProcAddress)
sl@0
   277
            {
sl@0
   278
            return KErrNotSupported;
sl@0
   279
            }
sl@0
   280
	    }
sl@0
   281
sl@0
   282
	return  KErrNone;
sl@0
   283
	}
sl@0
   284
sl@0
   285
TFuncPtrEglProc CEglDriver::GetProcAddress(const char* aName) const
sl@0
   286
   {
sl@0
   287
   // start with EGL extension functions
sl@0
   288
   //
sl@0
   289
   TPtrC8 procName(reinterpret_cast<const TUint8*>(aName));
sl@0
   290
   const TInt nExts = sizeof(KEglExtensionFuncs) / sizeof(KEglExtensionFuncs[0]);
sl@0
   291
   for (TInt idx=0; idx<nExts; ++idx)
sl@0
   292
       {
sl@0
   293
       if (procName == TPtrC8(reinterpret_cast<TUint8*>(KEglExtensionFuncs[idx].iName)))
sl@0
   294
           {
sl@0
   295
           return KEglExtensionFuncs[idx].iProc;
sl@0
   296
           }
sl@0
   297
       }
sl@0
   298
   
sl@0
   299
   // not an EGL extension, pass it on to OWF
sl@0
   300
   //
sl@0
   301
   return iOwfFuncTable.iOwfGetProcAddress ? iOwfFuncTable.iOwfGetProcAddress(aName) : NULL;
sl@0
   302
   }