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"
18 const TInt KEglMajorVersion = 1;
19 const TInt KEglMinorVersion = 4;
21 #define KEglClientApis ""
22 #define KEglVendor "Nokia"
23 #define KEglVersion "1.4 Reference EGL"
24 #define KEglExtensions "EGL_KHR_reusable_sync" /* additonal extensions should be added beginning with a space */ \
25 " EGL_NOK__private__signal_sync"
27 // Helper macros for repetitive task
29 #define CHECK_FOR_NULL_DISPLAY(handle, retval) \
30 if (handle == EGL_NO_DISPLAY) \
32 SetError(EGL_BAD_DISPLAY); \
36 #define CHECK_FOR_NULL_SYNCOBJ(handle, retval) \
37 if (handle == EGL_NO_SYNC_KHR) \
39 SetError(EGL_BAD_PARAMETER); \
43 #define GET_DISPLAY_RETURN_IF_ERR(drv, display, handle, retval) \
44 CEglDisplay* display = iDriver.FindDisplay(handle); \
48 SetError(EGL_BAD_DISPLAY); \
51 if (!display->IsInitialized()) \
54 SetError(EGL_NOT_INITIALIZED); \
58 #define GET_SYNCOBJ_RETURN_IF_ERR(drv, display, sync, handle, retval) \
59 CEglSync* sync = display->FindSyncObj(handle); \
63 SetError(EGL_BAD_PARAMETER); \
66 if (sync->IsDestroyed()) /* sync obj has been marked as destroyed */ \
69 SetError(EGL_BAD_PARAMETER); \
73 CEglThreadSession::CEglThreadSession(CEglDriver& aDriver):
79 CEglThreadSession::~CEglThreadSession()
84 CEglThreadSession* CEglThreadSession::Static()
86 CEglThreadSession* es = reinterpret_cast<CEglThreadSession*>(Dll::Tls());
92 const TInt err = CEglDriver::Open();
98 // CEglDriver is reference counted. As we successfuly open the driver, pls.iDriver will be non-null
99 // and it should be safe to cache the pointer inside CEglThreadSession
100 CEglDriver* drv = CEglDriver::GetDriver();
101 __ASSERT_DEBUG(drv, User::Panic(KEglPanicCategory, EEglPanicDriverNull));
103 // create session object on default thread's heap
104 es = new CEglThreadSession(*drv);
105 if (!es || Dll::SetTls(es)!= KErrNone)
114 void CEglThreadSession::SetError(EGLint aError)
116 // EGL spec section 3.1, GetError will return the status of the most recent EGL function call
117 // so we will always override the error status
121 EGLint CEglThreadSession::EglGetError()
123 // eglGetError always succeed so it will set error state to EGL_SUCCESS
124 const EGLint lastError = iError;
125 iError = EGL_SUCCESS;
129 EGLDisplay CEglThreadSession::EglGetDisplay(NativeDisplayType aDisplayId)
131 // EGL spec section 3.2: we do not need to raise EGL error when GetDisplay fails
132 SetError(EGL_SUCCESS);
134 if (aDisplayId != EGL_DEFAULT_DISPLAY)
136 return EGL_NO_DISPLAY;
139 // default display is created when driver is initialised the first time and will
140 // be destroyed when all threads within process have called eglReleaseThread
142 return KEglDefaultDisplayHandle;
145 EGLBoolean CEglThreadSession::EglInitialize(EGLDisplay aDisplay, EGLint* aMajor, EGLint* aMinor)
147 SetError(EGL_SUCCESS);
149 CHECK_FOR_NULL_DISPLAY(aDisplay, EGL_FALSE)
152 CEglDisplay* display = iDriver.FindDisplay(aDisplay);
156 SetError(EGL_BAD_DISPLAY);
160 const TInt err = display->Initialize();
165 SetError(EGL_NOT_INITIALIZED);
171 *aMajor = KEglMajorVersion;
175 *aMinor = KEglMinorVersion;
181 EGLBoolean CEglThreadSession::EglTerminate(EGLDisplay aDisplay)
183 SetError(EGL_SUCCESS);
185 CHECK_FOR_NULL_DISPLAY(aDisplay, EGL_FALSE)
188 CEglDisplay* display = iDriver.FindDisplay(aDisplay);
192 SetError(EGL_BAD_DISPLAY);
196 display->Terminate();
202 TFuncPtrEglProc CEglThreadSession::EglGetProcAddress(const char* aName)
204 SetError(EGL_SUCCESS);
211 // EGL spec does not mention about raising error if requested function not found
212 // This implementation does not set error and leave thread state unmodified
213 return iDriver.GetProcAddress(aName);
216 const char* CEglThreadSession::EglQueryString(EGLDisplay aDisplay, EGLint aName)
218 SetError(EGL_SUCCESS);
220 const char* str = NULL;
222 CHECK_FOR_NULL_DISPLAY(aDisplay, str)
225 GET_DISPLAY_RETURN_IF_ERR(iDriver, display, aDisplay, str)
230 case EGL_CLIENT_APIS:
231 str = KEglClientApis;
234 str = KEglExtensions;
243 SetError(EGL_BAD_PARAMETER);
250 EGLSyncKHR CEglThreadSession::EglCreateSyncKhr(EGLDisplay aDisplay, EGLenum aType, const EGLint *aAttribList)
252 SetError(EGL_SUCCESS);
254 CHECK_FOR_NULL_DISPLAY(aDisplay, EGL_NO_SYNC_KHR)
256 if (aType != EGL_SYNC_REUSABLE_KHR || (aAttribList && *aAttribList != EGL_NONE))
258 SetError(EGL_BAD_ATTRIBUTE);
259 return EGL_NO_SYNC_KHR;
263 GET_DISPLAY_RETURN_IF_ERR(iDriver, display, aDisplay, EGL_NO_SYNC_KHR)
265 CEglSync* syncObj = display->CreateSyncObj();
270 SetError(EGL_BAD_ALLOC);
271 return EGL_NO_SYNC_KHR;
274 return reinterpret_cast<EGLSyncKHR>(syncObj);
277 EGLBoolean CEglThreadSession::EglDestroySyncKhr(EGLDisplay aDisplay, EGLSyncKHR aSync)
279 SetError(EGL_SUCCESS);
281 CHECK_FOR_NULL_DISPLAY(aDisplay, EGL_FALSE)
282 CHECK_FOR_NULL_SYNCOBJ(aSync, EGL_FALSE)
285 GET_DISPLAY_RETURN_IF_ERR(iDriver, display, aDisplay, EGL_FALSE)
287 const TInt err = display->DestroySyncObj(aSync);
292 SetError(EGL_BAD_PARAMETER);
299 EGLint CEglThreadSession::EglClientWaitSyncKhr(EGLDisplay aDisplay, EGLSyncKHR aSync, EGLint aFlags, EGLTimeKHR aTimeout)
301 SetError(EGL_SUCCESS);
303 CHECK_FOR_NULL_DISPLAY(aDisplay, EGL_FALSE)
304 CHECK_FOR_NULL_SYNCOBJ(aSync, EGL_FALSE)
306 const EGLint supportedFlags = EGL_SYNC_FLUSH_COMMANDS_BIT_KHR;
307 if (aFlags & ~supportedFlags)
309 SetError(EGL_BAD_PARAMETER);
314 GET_DISPLAY_RETURN_IF_ERR(iDriver, display, aDisplay, EGL_FALSE)
315 GET_SYNCOBJ_RETURN_IF_ERR(iDriver, display, syncObj, aSync, EGL_FALSE)
317 // increment refcount to mark this sync obj in use and prevent it from being destroyed when other thread calls eglDestroySyncKHR or eglTerminate
320 // release display lock as we're going to wait on sync object after this point, not releasing display lock at this
321 // point will cause deadlock
325 // sync obj refcount has been incremented so it won't get destroyed even if other thread call eglDestroySyncKHR or eglTerminate
328 // we do not support client apis, so flushing flags will be ignored in this implementation
329 EGLint err = syncObj->Wait(aTimeout);
331 // decrement refcount
332 // sync obj will be destroyted if refcount is 0 e.g. other thread issues eglDestroySyncKHR or eglTerminate while this thread
338 // we do not check error here as it is passed back to caller
343 EGLBoolean CEglThreadSession::EglSignalSyncKhr(EGLDisplay aDisplay, EGLSyncKHR aSync, EGLenum aMode)
345 const EGLint err = EglSignalSyncInternal(aDisplay, aSync, aMode);
347 return err == EGL_SUCCESS;
350 EGLint CEglThreadSession::EglSignalSyncInternal(EGLDisplay aDisplay, EGLSyncKHR aSync, EGLenum aMode)
352 if (aDisplay == EGL_NO_DISPLAY)
354 return EGL_BAD_DISPLAY;
356 if (aSync == EGL_NO_SYNC_KHR)
358 return EGL_BAD_PARAMETER;
360 if (aMode != EGL_SIGNALED_KHR && aMode != EGL_UNSIGNALED_KHR)
362 return EGL_BAD_PARAMETER;
367 CEglDisplay* display = iDriver.FindDisplay(aDisplay);
371 return EGL_BAD_DISPLAY;
373 if (!display->IsInitialized())
376 return EGL_NOT_INITIALIZED;
379 CEglSync* syncObj = display->FindSyncObj(aSync);
383 return EGL_BAD_PARAMETER;
385 if (syncObj->Type() != EGL_SYNC_REUSABLE_KHR)
388 return EGL_BAD_MATCH;
390 if (syncObj->IsDestroyed()) /* sync obj has been marked as destroyed */
393 return EGL_BAD_PARAMETER;
396 syncObj->Signal(aMode);
402 EGLBoolean CEglThreadSession::EglGetSyncAttribKhr(EGLDisplay aDisplay, EGLSyncKHR aSync, EGLint aAttribute, EGLint *aValue)
404 SetError(EGL_SUCCESS);
406 CHECK_FOR_NULL_DISPLAY(aDisplay, EGL_FALSE)
407 CHECK_FOR_NULL_SYNCOBJ(aSync, EGL_FALSE)
409 if (aAttribute != EGL_SYNC_TYPE_KHR && aAttribute != EGL_SYNC_STATUS_KHR)
411 SetError(EGL_BAD_ATTRIBUTE);
417 SetError(EGL_BAD_PARAMETER);
423 GET_DISPLAY_RETURN_IF_ERR(iDriver, display, aDisplay, EGL_FALSE)
424 GET_SYNCOBJ_RETURN_IF_ERR(iDriver, display, syncObj, aSync, EGL_FALSE)
426 if (syncObj->Type() != EGL_SYNC_REUSABLE_KHR)
429 SetError(EGL_BAD_MATCH);
435 case EGL_SYNC_TYPE_KHR:
436 *aValue = syncObj->Type();
439 case EGL_SYNC_STATUS_KHR:
440 *aValue = syncObj->Status();
450 void CEglThreadSession::EglHeapMarkStart()
453 iDriver.Heap().__DbgMarkStart();
457 EGLint CEglThreadSession::EglHeapMarkEnd(EGLint aCount)
460 const TInt cell = iDriver.Heap().__DbgMarkEnd(aCount);
466 void CEglThreadSession::EglHeapSetBurstAllocFail(EGLenum aType, EGLint aRate, EGLint aBurst)
469 iDriver.Heap().__DbgSetBurstAllocFail(static_cast<RAllocator::TAllocFail>(aType), aRate, aBurst);