os/graphics/egl/eglrefimpl/src/session.cpp
author sl
Tue, 10 Jun 2014 14:32:02 +0200
changeset 1 260cb5ec6c19
permissions -rw-r--r--
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".
     7 //
     8 // Initial Contributors:
     9 // Nokia Corporation - initial contribution.
    10 //
    11 // Contributors:
    12 //
    13 // Description:
    14 // Reference EGL implementation to support EGL sync objects and OpenWF extensions
    15 
    16 #include "eglprivate.h"
    17 
    18 const TInt KEglMajorVersion = 1;
    19 const TInt KEglMinorVersion = 4;
    20 
    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"
    26 
    27 // Helper macros for repetitive task
    28 //
    29 #define CHECK_FOR_NULL_DISPLAY(handle, retval) \
    30     if (handle == EGL_NO_DISPLAY) \
    31         { \
    32         SetError(EGL_BAD_DISPLAY); \
    33         return retval; \
    34         }
    35 
    36 #define CHECK_FOR_NULL_SYNCOBJ(handle, retval) \
    37     if (handle == EGL_NO_SYNC_KHR) \
    38         { \
    39         SetError(EGL_BAD_PARAMETER); \
    40         return retval; \
    41         }
    42 
    43 #define GET_DISPLAY_RETURN_IF_ERR(drv, display, handle, retval) \
    44     CEglDisplay* display = iDriver.FindDisplay(handle); \
    45     if (!display) \
    46         { \
    47         drv.Unlock(); \
    48         SetError(EGL_BAD_DISPLAY); \
    49         return retval; \
    50         } \
    51     if (!display->IsInitialized()) \
    52         { \
    53         drv.Unlock(); \
    54         SetError(EGL_NOT_INITIALIZED); \
    55         return retval; \
    56         }
    57 
    58 #define GET_SYNCOBJ_RETURN_IF_ERR(drv, display, sync, handle, retval) \
    59     CEglSync* sync = display->FindSyncObj(handle); \
    60     if (!sync) \
    61         { \
    62         drv.Unlock(); \
    63         SetError(EGL_BAD_PARAMETER); \
    64         return retval; \
    65         } \
    66     if (sync->IsDestroyed()) /* sync obj has been marked as destroyed */ \
    67         { \
    68         drv.Unlock(); \
    69         SetError(EGL_BAD_PARAMETER); \
    70         return retval; \
    71         }
    72                                                                 
    73 CEglThreadSession::CEglThreadSession(CEglDriver& aDriver):
    74 	iDriver(aDriver),
    75 	iError(EGL_SUCCESS)
    76 	{
    77 	}
    78 
    79 CEglThreadSession::~CEglThreadSession()
    80 	{
    81 	CEglDriver::Close();
    82 	}
    83 
    84 CEglThreadSession* CEglThreadSession::Static()
    85 	{
    86 	CEglThreadSession* es = reinterpret_cast<CEglThreadSession*>(Dll::Tls());
    87 	if (es)
    88 		{
    89 		return es;
    90 		}
    91 
    92 	const TInt err = CEglDriver::Open();
    93 	if (err != KErrNone)
    94 		{
    95 		return NULL;
    96 		}
    97 
    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));
   102 
   103 	// create session object on default thread's heap
   104 	es = new CEglThreadSession(*drv);
   105 	if (!es || Dll::SetTls(es)!= KErrNone)
   106 		{
   107 		delete es;
   108 		return NULL;
   109 		}
   110 	
   111 	return es;
   112 	}
   113 
   114 void CEglThreadSession::SetError(EGLint aError)
   115 	{
   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
   118 	iError = aError;
   119 	}
   120 
   121 EGLint CEglThreadSession::EglGetError()
   122 	{
   123 	// eglGetError always succeed so it will set error state to EGL_SUCCESS
   124 	const EGLint lastError = iError;
   125 	iError = EGL_SUCCESS;
   126 	return lastError;
   127 	}
   128 
   129 EGLDisplay CEglThreadSession::EglGetDisplay(NativeDisplayType aDisplayId)
   130     {
   131     // EGL spec section 3.2: we do not need to raise EGL error when GetDisplay fails
   132     SetError(EGL_SUCCESS);
   133 
   134     if (aDisplayId != EGL_DEFAULT_DISPLAY)
   135         {
   136         return EGL_NO_DISPLAY;
   137         }
   138 
   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
   141 
   142     return KEglDefaultDisplayHandle;
   143     }
   144 
   145 EGLBoolean CEglThreadSession::EglInitialize(EGLDisplay aDisplay, EGLint* aMajor, EGLint* aMinor)
   146     {
   147     SetError(EGL_SUCCESS);
   148     
   149     CHECK_FOR_NULL_DISPLAY(aDisplay, EGL_FALSE)
   150 
   151     iDriver.Lock();
   152     CEglDisplay* display = iDriver.FindDisplay(aDisplay);
   153     if (!display)
   154         {
   155         iDriver.Unlock();
   156         SetError(EGL_BAD_DISPLAY);
   157         return EGL_FALSE;
   158         }
   159 
   160     const TInt err = display->Initialize();
   161     iDriver.Unlock();
   162     
   163     if (err != KErrNone)
   164         {
   165         SetError(EGL_NOT_INITIALIZED);
   166         return EGL_FALSE;
   167         }
   168 
   169     if (aMajor)
   170         {
   171         *aMajor = KEglMajorVersion;
   172         }
   173     if (aMinor)
   174         {
   175         *aMinor = KEglMinorVersion;
   176         }
   177 
   178     return EGL_TRUE;
   179     }
   180 
   181 EGLBoolean CEglThreadSession::EglTerminate(EGLDisplay aDisplay)
   182     {
   183     SetError(EGL_SUCCESS);
   184     
   185     CHECK_FOR_NULL_DISPLAY(aDisplay, EGL_FALSE)
   186 
   187     iDriver.Lock();
   188     CEglDisplay* display = iDriver.FindDisplay(aDisplay);
   189     if (!display)
   190         {
   191         iDriver.Unlock();
   192         SetError(EGL_BAD_DISPLAY);
   193         return EGL_FALSE;
   194         }
   195 
   196     display->Terminate();
   197     iDriver.Unlock();
   198     
   199     return EGL_TRUE;
   200     }
   201 
   202 TFuncPtrEglProc CEglThreadSession::EglGetProcAddress(const char* aName)
   203     {
   204     SetError(EGL_SUCCESS);
   205         
   206     if(!aName)
   207         {
   208         return NULL;
   209         }
   210     
   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);
   214     }
   215 
   216 const char* CEglThreadSession::EglQueryString(EGLDisplay aDisplay, EGLint aName)
   217     {
   218     SetError(EGL_SUCCESS);
   219     
   220     const char* str = NULL;
   221 
   222     CHECK_FOR_NULL_DISPLAY(aDisplay, str)
   223 
   224     iDriver.Lock();
   225     GET_DISPLAY_RETURN_IF_ERR(iDriver, display, aDisplay, str)
   226     iDriver.Unlock();
   227 
   228     switch (aName)
   229         {
   230         case EGL_CLIENT_APIS:
   231             str = KEglClientApis;
   232             break;
   233         case EGL_EXTENSIONS:
   234             str = KEglExtensions;
   235             break;
   236         case EGL_VENDOR:
   237             str = KEglVendor;
   238             break;
   239         case EGL_VERSION:
   240             str = KEglVersion;
   241             break;
   242         default:
   243             SetError(EGL_BAD_PARAMETER);
   244             break;
   245         }
   246 
   247     return str;
   248     }
   249 
   250 EGLSyncKHR CEglThreadSession::EglCreateSyncKhr(EGLDisplay aDisplay, EGLenum aType, const EGLint *aAttribList)
   251     {
   252     SetError(EGL_SUCCESS);
   253     
   254     CHECK_FOR_NULL_DISPLAY(aDisplay, EGL_NO_SYNC_KHR)
   255 
   256     if (aType != EGL_SYNC_REUSABLE_KHR || (aAttribList && *aAttribList != EGL_NONE))
   257         {
   258         SetError(EGL_BAD_ATTRIBUTE);
   259         return EGL_NO_SYNC_KHR;
   260         }
   261     
   262     iDriver.Lock();
   263     GET_DISPLAY_RETURN_IF_ERR(iDriver, display, aDisplay, EGL_NO_SYNC_KHR)
   264     
   265     CEglSync* syncObj = display->CreateSyncObj();
   266     iDriver.Unlock();
   267     
   268     if (!syncObj)
   269         {
   270         SetError(EGL_BAD_ALLOC);
   271         return EGL_NO_SYNC_KHR;
   272         }
   273 
   274     return reinterpret_cast<EGLSyncKHR>(syncObj);
   275     }
   276 
   277 EGLBoolean CEglThreadSession::EglDestroySyncKhr(EGLDisplay aDisplay, EGLSyncKHR aSync)
   278     {
   279     SetError(EGL_SUCCESS);
   280     
   281     CHECK_FOR_NULL_DISPLAY(aDisplay, EGL_FALSE)
   282     CHECK_FOR_NULL_SYNCOBJ(aSync, EGL_FALSE)
   283 
   284     iDriver.Lock();
   285     GET_DISPLAY_RETURN_IF_ERR(iDriver, display, aDisplay, EGL_FALSE)
   286     
   287     const TInt err = display->DestroySyncObj(aSync);
   288     iDriver.Unlock();
   289 
   290     if (err != KErrNone)
   291         {
   292         SetError(EGL_BAD_PARAMETER);
   293         return EGL_FALSE;
   294         }
   295 
   296     return EGL_TRUE;
   297     }
   298 
   299 EGLint CEglThreadSession::EglClientWaitSyncKhr(EGLDisplay aDisplay, EGLSyncKHR aSync, EGLint aFlags, EGLTimeKHR aTimeout)
   300     {
   301     SetError(EGL_SUCCESS);
   302     
   303     CHECK_FOR_NULL_DISPLAY(aDisplay, EGL_FALSE)
   304     CHECK_FOR_NULL_SYNCOBJ(aSync, EGL_FALSE)
   305     
   306     const EGLint supportedFlags = EGL_SYNC_FLUSH_COMMANDS_BIT_KHR;
   307     if (aFlags & ~supportedFlags)
   308         {
   309         SetError(EGL_BAD_PARAMETER);
   310         return EGL_FALSE;
   311         }
   312     
   313     iDriver.Lock();
   314     GET_DISPLAY_RETURN_IF_ERR(iDriver, display, aDisplay, EGL_FALSE)
   315     GET_SYNCOBJ_RETURN_IF_ERR(iDriver, display, syncObj, aSync, EGL_FALSE)
   316 
   317     // increment refcount to mark this sync obj in use and prevent it from being destroyed when other thread calls eglDestroySyncKHR or eglTerminate
   318     syncObj->Open();
   319     
   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
   322     // coverity[stale]
   323     iDriver.Unlock();
   324 
   325     // sync obj refcount has been incremented so it won't get destroyed even if other thread call eglDestroySyncKHR or eglTerminate
   326     // at this point
   327 
   328     // we do not support client apis, so flushing flags will be ignored in this implementation
   329     EGLint err = syncObj->Wait(aTimeout);
   330 
   331     // decrement refcount
   332     // sync obj will be destroyted if refcount is 0 e.g. other thread issues eglDestroySyncKHR or eglTerminate while this thread
   333     // is waiting on it
   334     iDriver.Lock();
   335     syncObj->Close();
   336     iDriver.Unlock();
   337 
   338     // we do not check error here as it is passed back to caller
   339 
   340     return err;
   341     }
   342 
   343 EGLBoolean CEglThreadSession::EglSignalSyncKhr(EGLDisplay aDisplay, EGLSyncKHR aSync, EGLenum aMode)
   344     {
   345     const EGLint err = EglSignalSyncInternal(aDisplay, aSync, aMode);
   346     SetError(err);
   347     return err == EGL_SUCCESS;
   348     }
   349 
   350 EGLint CEglThreadSession::EglSignalSyncInternal(EGLDisplay aDisplay, EGLSyncKHR aSync, EGLenum aMode)
   351     {
   352     if (aDisplay == EGL_NO_DISPLAY)
   353         {
   354         return EGL_BAD_DISPLAY;
   355         }
   356     if (aSync == EGL_NO_SYNC_KHR)
   357         {
   358         return EGL_BAD_PARAMETER;
   359         }
   360     if (aMode != EGL_SIGNALED_KHR && aMode != EGL_UNSIGNALED_KHR)
   361         {
   362         return EGL_BAD_PARAMETER;
   363         }
   364 
   365     iDriver.Lock();
   366 
   367     CEglDisplay* display = iDriver.FindDisplay(aDisplay);
   368     if (!display)
   369         {
   370         iDriver.Unlock();
   371         return EGL_BAD_DISPLAY;
   372         }
   373     if (!display->IsInitialized())
   374         {
   375         iDriver.Unlock();
   376         return EGL_NOT_INITIALIZED;
   377         }
   378 
   379     CEglSync* syncObj = display->FindSyncObj(aSync);
   380     if (!syncObj)
   381         {
   382         iDriver.Unlock();
   383         return EGL_BAD_PARAMETER;
   384         }
   385     if (syncObj->Type() != EGL_SYNC_REUSABLE_KHR)
   386         {
   387         iDriver.Unlock();
   388         return EGL_BAD_MATCH;
   389         }
   390     if (syncObj->IsDestroyed()) /* sync obj has been marked as destroyed */
   391         {
   392         iDriver.Unlock();
   393         return EGL_BAD_PARAMETER;
   394         }
   395     
   396     syncObj->Signal(aMode);
   397     iDriver.Unlock();
   398     
   399     return EGL_SUCCESS;
   400     }
   401 
   402 EGLBoolean CEglThreadSession::EglGetSyncAttribKhr(EGLDisplay aDisplay, EGLSyncKHR aSync, EGLint aAttribute, EGLint *aValue)
   403     {
   404     SetError(EGL_SUCCESS);
   405     
   406     CHECK_FOR_NULL_DISPLAY(aDisplay, EGL_FALSE)
   407     CHECK_FOR_NULL_SYNCOBJ(aSync, EGL_FALSE)
   408     
   409     if (aAttribute != EGL_SYNC_TYPE_KHR && aAttribute != EGL_SYNC_STATUS_KHR)
   410         {
   411         SetError(EGL_BAD_ATTRIBUTE);
   412         return EGL_FALSE;
   413         }
   414 
   415     if (!aValue)
   416         {
   417         SetError(EGL_BAD_PARAMETER);
   418         return EGL_FALSE;
   419         }
   420 
   421     iDriver.Lock();
   422 
   423     GET_DISPLAY_RETURN_IF_ERR(iDriver, display, aDisplay, EGL_FALSE)
   424     GET_SYNCOBJ_RETURN_IF_ERR(iDriver, display, syncObj, aSync, EGL_FALSE)
   425 
   426     if (syncObj->Type() != EGL_SYNC_REUSABLE_KHR)
   427         {
   428         iDriver.Unlock();
   429         SetError(EGL_BAD_MATCH);
   430         return EGL_FALSE;
   431         }
   432     
   433     switch (aAttribute)
   434         {
   435         case EGL_SYNC_TYPE_KHR:
   436             *aValue = syncObj->Type();
   437             break;
   438             
   439         case EGL_SYNC_STATUS_KHR:
   440             *aValue = syncObj->Status();
   441             break;
   442         }
   443 
   444     iDriver.Unlock();
   445     
   446     return EGL_TRUE;
   447     }
   448 
   449 #ifdef _DEBUG
   450 void CEglThreadSession::EglHeapMarkStart()
   451     {
   452     iDriver.Lock();
   453     iDriver.Heap().__DbgMarkStart();
   454     iDriver.Unlock();
   455     }
   456 
   457 EGLint CEglThreadSession::EglHeapMarkEnd(EGLint aCount)
   458     {
   459     iDriver.Lock();
   460     const TInt cell = iDriver.Heap().__DbgMarkEnd(aCount);
   461     iDriver.Unlock();
   462     
   463     return cell;
   464     }
   465 
   466 void CEglThreadSession::EglHeapSetBurstAllocFail(EGLenum aType, EGLint aRate, EGLint aBurst)
   467     {
   468     iDriver.Lock();
   469     iDriver.Heap().__DbgSetBurstAllocFail(static_cast<RAllocator::TAllocFail>(aType), aRate, aBurst);
   470     iDriver.Unlock();
   471     }
   472 
   473 #endif //_DEBUG