os/graphics/egl/eglrefimpl/src/syncobj.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
sl@0
    18
CEglSync::CEglSync(CEglDisplay& aDisplay):
sl@0
    19
    iDisplay(aDisplay),
sl@0
    20
    iType(EGL_SYNC_REUSABLE_KHR),
sl@0
    21
    iStatus(EGL_UNSIGNALED_KHR)
sl@0
    22
    {
sl@0
    23
    }
sl@0
    24
sl@0
    25
CEglSync::~CEglSync()
sl@0
    26
    {
sl@0
    27
    iCondVar.Close();
sl@0
    28
    iMutex.Close();
sl@0
    29
    }
sl@0
    30
sl@0
    31
TInt CEglSync::Construct()
sl@0
    32
    {
sl@0
    33
    TInt err = iMutex.CreateLocal();
sl@0
    34
    if (err != KErrNone)
sl@0
    35
        {
sl@0
    36
        return err;
sl@0
    37
        }
sl@0
    38
sl@0
    39
    err = iCondVar.CreateLocal();
sl@0
    40
    if (err != KErrNone)
sl@0
    41
        {
sl@0
    42
        return err;
sl@0
    43
        }
sl@0
    44
    
sl@0
    45
    err = iDisplay.RegisterSyncObj(*this);
sl@0
    46
    if (err != KErrNone)
sl@0
    47
        {
sl@0
    48
        return err;
sl@0
    49
        }
sl@0
    50
sl@0
    51
    iRefCount = 1;
sl@0
    52
    return KErrNone;
sl@0
    53
    }
sl@0
    54
sl@0
    55
CEglSync* CEglSync::Create(CEglDisplay& aDisplay)
sl@0
    56
    {
sl@0
    57
    // sync object will be allocated in the EGL shared heap and added to sync object list
sl@0
    58
    // we need to switch current heap to EGL shared heap
sl@0
    59
    //
sl@0
    60
    RHeap* callerHeap = User::SwitchHeap(&aDisplay.Heap());
sl@0
    61
sl@0
    62
    CEglSync* syncObj = new CEglSync(aDisplay);
sl@0
    63
    if (!syncObj)
sl@0
    64
        {
sl@0
    65
        User::SwitchHeap(callerHeap);
sl@0
    66
        return NULL;
sl@0
    67
        }
sl@0
    68
    
sl@0
    69
    const TInt err = syncObj->Construct();
sl@0
    70
    if (err != KErrNone)
sl@0
    71
        {
sl@0
    72
        delete syncObj;
sl@0
    73
        User::SwitchHeap(callerHeap);
sl@0
    74
        return NULL;
sl@0
    75
        }
sl@0
    76
    
sl@0
    77
    User::SwitchHeap(callerHeap);
sl@0
    78
    return syncObj;
sl@0
    79
    }
sl@0
    80
sl@0
    81
void CEglSync::Destroy()
sl@0
    82
    {
sl@0
    83
    // multiple calls to Destroy() is not allowed, it's either coming from eglDestroySyncKHR or eglTerminate
sl@0
    84
    //
sl@0
    85
    __ASSERT_DEBUG(!iIsDestroyed, User::Panic(KEglPanicCategory, EEglPanicSyncObjHasBeenDestroyed));
sl@0
    86
    
sl@0
    87
    iIsDestroyed = ETrue;
sl@0
    88
sl@0
    89
    // wake up all waiting threads
sl@0
    90
    iCondVar.Broadcast();
sl@0
    91
sl@0
    92
    // always remove sync obj from hash map when it is destroyed, the actual deletion will be done from Close(),
sl@0
    93
    // which can happen when eglClientWaitSyncKHR is called by user
sl@0
    94
    RHeap* callerHeap = User::SwitchHeap(&iDisplay.Heap());
sl@0
    95
    iDisplay.UnregisterSyncObj(*this);
sl@0
    96
    User::SwitchHeap(callerHeap);
sl@0
    97
    
sl@0
    98
    // decrement refcount for this sync obj, it will delete the object if refcount is zero
sl@0
    99
    Close();
sl@0
   100
    }
sl@0
   101
sl@0
   102
void CEglSync::Close()
sl@0
   103
    {
sl@0
   104
    if (--iRefCount == 0)
sl@0
   105
        {
sl@0
   106
        // we're here either from Destroy() or eglClientWaitSyncKHR
sl@0
   107
        RHeap* callerHeap = User::SwitchHeap(&iDisplay.Heap());
sl@0
   108
        delete this;
sl@0
   109
        User::SwitchHeap(callerHeap);
sl@0
   110
        }
sl@0
   111
    }
sl@0
   112
sl@0
   113
void CEglSync::Signal(EGLenum aMode)
sl@0
   114
    {
sl@0
   115
    iMutex.Wait();
sl@0
   116
    if (iStatus != aMode)
sl@0
   117
        {
sl@0
   118
        iStatus = aMode;
sl@0
   119
        if (iStatus == EGL_SIGNALED_KHR)
sl@0
   120
            {
sl@0
   121
            iCondVar.Broadcast();
sl@0
   122
            }
sl@0
   123
        }
sl@0
   124
    iMutex.Signal();
sl@0
   125
    }
sl@0
   126
sl@0
   127
EGLint CEglSync::Wait(EGLTimeKHR aTimeOut)
sl@0
   128
    {
sl@0
   129
    // driver display lock is not held when we're about to enter block wait on condition variable
sl@0
   130
    // we use sync object mutex to synchronise threads access from this point until end of this function
sl@0
   131
    iMutex.Wait();
sl@0
   132
    EGLint errCode = EGL_CONDITION_SATISFIED_KHR;
sl@0
   133
    
sl@0
   134
    if (iStatus == EGL_UNSIGNALED_KHR)
sl@0
   135
        {
sl@0
   136
        switch(aTimeOut) 
sl@0
   137
            {
sl@0
   138
            case EGL_FOREVER_KHR:
sl@0
   139
                {
sl@0
   140
                const TInt res = iCondVar.Wait(iMutex); 
sl@0
   141
                //we do not expect to fail here
sl@0
   142
                __ASSERT_DEBUG(res == KErrNone, User::Panic(KEglPanicCategory, EEglPanicCondVarWaitFail));
sl@0
   143
                break;
sl@0
   144
                }
sl@0
   145
            case 0:
sl@0
   146
                {
sl@0
   147
                //by setting this we notify the caller that the sync object is in unsignaled state
sl@0
   148
                errCode = EGL_TIMEOUT_EXPIRED_KHR; 
sl@0
   149
                break;
sl@0
   150
                }
sl@0
   151
            default:
sl@0
   152
                {
sl@0
   153
                // Since the supported range of timeout at function RCondVar::TimedWait(mutex, timeout) 
sl@0
   154
                // is 0 to KMaxTInt, looping mechanism below is used to support 64bit timeout.      
sl@0
   155
                //
sl@0
   156
                TInt res = KErrTimedOut;
sl@0
   157
                for(TInt64 timeoutMicroseconds = aTimeOut/1000; (res == KErrTimedOut) && (timeoutMicroseconds > 0); timeoutMicroseconds -= KMaxTInt)
sl@0
   158
                    {
sl@0
   159
                    res = iCondVar.TimedWait(iMutex, (timeoutMicroseconds > KMaxTInt?KMaxTInt:timeoutMicroseconds));
sl@0
   160
                    //we do not expect to fail here
sl@0
   161
                    __ASSERT_DEBUG(res == KErrNone || res == KErrTimedOut, User::Panic(KEglPanicCategory, EEglPanicCondVarTimedWaitFail));
sl@0
   162
                    }
sl@0
   163
                if(res == KErrTimedOut)
sl@0
   164
                    {
sl@0
   165
                    errCode = EGL_TIMEOUT_EXPIRED_KHR;
sl@0
   166
                    }
sl@0
   167
                break;
sl@0
   168
                }
sl@0
   169
            }
sl@0
   170
        }
sl@0
   171
sl@0
   172
    iMutex.Signal();
sl@0
   173
    return errCode;
sl@0
   174
    }