Update contrib.
1 // Copyright (c) 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 // Reference EGL implementation to support EGL sync objects and OpenWF extensions
16 #include "eglprivate.h"
17 #include <EGL/egluids.hrh>
19 #include <WF/openwfcuids.hrh>
21 // OpenWF Composition library
23 _LIT(KOwfDllName, "libWFC.dll");
24 const TUid KOwfDllUid1 = {KDynamicLibraryUidValue};
25 const TUid KOwfDllUid2 = {KUidSharedDllUidValue};
26 const TUid KOwfDllUid3 = {KUidOpenWfcDllUidValue};
27 const TUidType KOwfDllUidType = TUidType(KOwfDllUid1, KOwfDllUid2, KOwfDllUid3);
29 // EGL_KHR_reusable_sync extensions API
30 // Declare here so their addresses can be used to build extensions table
32 EGLSyncKHR eglCreateSyncKHR(EGLDisplay aDisplay, EGLenum aType, const EGLint *aAttribList);
33 EGLBoolean eglDestroySyncKHR(EGLDisplay aDisplay, EGLSyncKHR aSync);
34 EGLint eglClientWaitSyncKHR(EGLDisplay aDisplay, EGLSyncKHR aSync, EGLint aFlags, EGLTimeKHR aTimeout);
35 EGLBoolean eglSignalSyncKHR(EGLDisplay aDisplay, EGLSyncKHR aSync, EGLenum aMode);
36 EGLBoolean eglGetSyncAttribKHR(EGLDisplay aDisplay, EGLSyncKHR aSync, EGLint aAttribute, EGLint *aValue);
40 EGLint egl_Private_SignalSyncNOK(EGLDisplay aDisplay, EGLSyncKHR aSync, EGLenum aMode);
42 // Extensions for shared heap OOM test
44 void egliDebugHeapMarkStart();
45 EGLint egliDebugHeapMarkEnd(EGLint aCount);
46 void egliDebugSetBurstAllocFail(EGLenum aType, EGLint aRate, EGLint aBurst);
49 struct TEglNameProcRecord
52 TFuncPtrEglProc iProc;
55 #define MAKE_NAME_PROC_RECORD(f) {#f,reinterpret_cast<TFuncPtrEglProc>(f)}
57 const TEglNameProcRecord KEglExtensionFuncs[] = {
58 MAKE_NAME_PROC_RECORD(eglCreateSyncKHR),
59 MAKE_NAME_PROC_RECORD(eglDestroySyncKHR),
60 MAKE_NAME_PROC_RECORD(eglClientWaitSyncKHR),
61 MAKE_NAME_PROC_RECORD(eglSignalSyncKHR),
62 MAKE_NAME_PROC_RECORD(eglGetSyncAttribKHR),
63 MAKE_NAME_PROC_RECORD(egl_Private_SignalSyncNOK)
66 MAKE_NAME_PROC_RECORD(egliDebugHeapMarkStart),
67 MAKE_NAME_PROC_RECORD(egliDebugHeapMarkEnd),
68 MAKE_NAME_PROC_RECORD(egliDebugSetBurstAllocFail)
72 // Use common EglPls() function to hide difference between WINS and target in dealing with PLS
76 static inline XEglPls* EglPls()
78 const TUid KUidEglDll = {KUidEGLDllUidValue};
79 return Pls<XEglPls>(KUidEglDll);
82 static XEglPls TheEglPls;
83 static inline XEglPls* EglPls()
90 // Macros for placement new and delete of shared objects
92 #define EGL_PLACEMENT_NEW(heap, obj, klass) klass* obj = NULL; \
93 TAny* buf = (heap).AllocZ(sizeof(klass)); \
96 obj = new(buf) klass((heap)); \
99 #define EGL_PLACEMENT_DELETE(heap, obj, klass) obj->~klass(); \
102 ///////////////////////////////////////////////////////////////////////////////
104 ///////////////////////////////////////////////////////////////////////////////
108 iError = iLock.CreateLocal();
117 ///////////////////////////////////////////////////////////////////////////////
119 ///////////////////////////////////////////////////////////////////////////////
121 CEglDriver::CEglDriver(RHeap& aHeap):
126 CEglDriver::~CEglDriver()
128 // EGL display was allocated on shared heap using placement new
131 EGL_PLACEMENT_DELETE(iHeap, iDisplay, CEglDisplay)
138 TInt CEglDriver::Open()
140 // we're in trouble if mutex creation failed during DLL initialisation
141 XEglPls& pls = *EglPls();
142 if (pls.iError != KErrNone)
151 ++(pls.iDriver->iRefCount);
156 // create shared heap in a local chunk to allow access to shared objects from any threads within Driver
157 // allow heap to grow from 256 bytes up to 1MB
158 const TInt KMaxHeapSize = 0x100000;
159 const TInt KWordAlign = 4;
160 RHeap* heap = User::ChunkHeap(NULL, KMinHeapSize, KMaxHeapSize, KMinHeapGrowBy, KWordAlign);
167 EGL_PLACEMENT_NEW(*heap, drv, CEglDriver)
175 const TInt err = drv->Construct();
178 EGL_PLACEMENT_DELETE(*heap, drv, CEglDriver)
186 pls.iDriver->iRefCount = 1;
193 void CEglDriver::Close()
195 XEglPls& pls = *EglPls();
196 // must never happen if CEglDriver::Open() succeed
197 __ASSERT_DEBUG(pls.iError == KErrNone, User::Panic(KEglPanicCategory, EEglPanicPlsMutexError));
203 if (--(pls.iDriver->iRefCount) == 0)
205 // copy shared heap pointer out as we're about to destroy the driver
206 RHeap& heap = pls.iDriver->iHeap;
208 // driver was allocated on shared heap using placement new
209 EGL_PLACEMENT_DELETE(heap, pls.iDriver, CEglDriver)
218 CEglDriver* CEglDriver::GetDriver()
220 XEglPls& pls = *EglPls();
224 // called when lock is held
226 TInt CEglDriver::Construct()
228 TInt err = iLock.CreateLocal();
234 // create default EGL display as part of driver initialisation, it will be destroyed when the driver is destroyed
235 // and recreated when driver is reinitialised
236 EGL_PLACEMENT_NEW(iHeap, disp, CEglDisplay)
244 // for security reason, use TUidType when loading OWF library to make sure it is not a spoofed DLL with the same name
245 err = iOwfLib.Load(KOwfDllName, KOwfDllUidType);
246 // allow initialisation to proceed even if OWF library is not found
249 // promote library handle from thread-relative to process-wide to allow closing from any thread
250 RLibrary dup = iOwfLib;
251 err = iOwfLib.Duplicate(RThread());
256 // if Duplicate succeed, it will overwrite iOwfLib with process-wide handle
257 // we need to close the original thread-relative handle
260 // In SymbianOS implementation of Khronos API, ordinal #1 is always reserved for internal implementation use as defined in
261 // SymbianOS standard DEF files for that API. This DEF file ensures BC among different implementations of that API.
262 // In OWF case, ordinal #1 refers to GetExtensionFunctionTable, a function which will return a table of function pointer.
263 // One of them is a function that implement GetProcAddress to retrieve OWF extension functions.
265 const TInt KOwfGetFuncTableOrdinalNum = 1;
266 typedef void (*TFuncPtrOwfGetFuncTable)(TOwfFuncTable*);
268 TFuncPtrOwfGetFuncTable owfGetFuncTable = reinterpret_cast<TFuncPtrOwfGetFuncTable>(iOwfLib.Lookup(KOwfGetFuncTableOrdinalNum));
269 if (!owfGetFuncTable)
274 owfGetFuncTable(&iOwfFuncTable);
276 if (!iOwfFuncTable.iOwfGetProcAddress)
278 return KErrNotSupported;
285 TFuncPtrEglProc CEglDriver::GetProcAddress(const char* aName) const
287 // start with EGL extension functions
289 TPtrC8 procName(reinterpret_cast<const TUint8*>(aName));
290 const TInt nExts = sizeof(KEglExtensionFuncs) / sizeof(KEglExtensionFuncs[0]);
291 for (TInt idx=0; idx<nExts; ++idx)
293 if (procName == TPtrC8(reinterpret_cast<TUint8*>(KEglExtensionFuncs[idx].iName)))
295 return KEglExtensionFuncs[idx].iProc;
299 // not an EGL extension, pass it on to OWF
301 return iOwfFuncTable.iOwfGetProcAddress ? iOwfFuncTable.iOwfGetProcAddress(aName) : NULL;