author | Faisal Memon <faisal.memon@nokia.com> |
Mon, 20 Sep 2010 14:29:05 +0100 | |
branch | bug235_bringup_0 |
changeset 24 | a3f46bb01be2 |
parent 17 | c9d1eb0bfac8 |
permissions | -rwxr-xr-x |
13 | 1 |
// Copyright (c) 2010 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 |
// Implementation of SgImage functions exported through eglGetProcAddress |
|
15 |
||
16 |
||
17 |
#include "eglapi.h" |
|
18 |
||
19 |
||
20 |
//////////////////////////////////////////////////////////////////////////////////////////////////// |
|
21 |
// CEgLImage |
|
22 |
//////////////////////////////////////////////////////////////////////////////////////////////////// |
|
23 |
||
24 |
#ifdef FAISALMEMON_S4_SGIMAGE |
|
25 |
CEglImage::CEglImage() : iSgHandle(), iDisplay(0), iSgImageInfo(), iPbufferHandle(0), iVgHandle(VG_INVALID_HANDLE), |
|
26 |
iCreateCount(1), iOpenCount(0), iIsDestroyed(EFalse) |
|
27 |
{ |
|
28 |
EGL_TRACE("CEglImage::CEglImage(0x%x)", this); |
|
29 |
} |
|
30 |
||
31 |
CEglImage::~CEglImage() |
|
32 |
{ |
|
33 |
EGL_TRACE("CEglImage::~CEglImage(0x%x) iCreateCount=%d, iOpenCount=%d", this, iCreateCount, iOpenCount); |
|
34 |
EGLPANIC_ASSERT_DEBUG( (iCreateCount == 0) && (iOpenCount == 0), EEglPanicEglImageRefCountNonZero); |
|
35 |
iSgHandle.Close(); |
|
36 |
} |
|
37 |
||
38 |
// initialise member data |
|
39 |
void CEglImage::Create(TSgDrawableId aSgImageId, EGLDisplay aDisplay, TSgImageInfo& aSgImageInfo, TInt aPbufferHandle, |
|
40 |
VGHandle aVgHandle) |
|
41 |
{ |
|
42 |
EGL_TRACE("CEglImage::Create iCreateCount=%d, iOpenCount=%d", iCreateCount, iOpenCount); |
|
43 |
EGLPANIC_ASSERT_DEBUG(iCreateCount == 0, EEglPanicEglImageRefCountNonZero); |
|
44 |
EGLPANIC_ASSERT_DEBUG(!iIsDestroyed, EEglPanicEglImageIsDestroyed); |
|
45 |
EGLPANIC_ASSERT_DEBUG(aSgImageId != KSgNullDrawableId, EEglPanicTemp); |
|
46 |
++iCreateCount; |
|
47 |
TInt err = iSgHandle.Open(aSgImageId); |
|
48 |
EGLPANIC_ASSERT_DEBUG(err == KErrNone, EEglPanicTemp); |
|
49 |
iDisplay = aDisplay; |
|
50 |
iSgImageInfo = aSgImageInfo; |
|
51 |
iVgHandle = aVgHandle; |
|
52 |
iPbufferHandle = aPbufferHandle; |
|
53 |
} |
|
54 |
||
55 |
// Duplicate open |
|
56 |
void CEglImage::Duplicate() |
|
57 |
{ |
|
58 |
EGL_TRACE("CEglImage::Duplicate iCreateCount=%d, iOpenCount=%d", iCreateCount, iOpenCount); |
|
59 |
EGLPANIC_ASSERT_DEBUG( (iCreateCount > 0) && !iIsDestroyed, EEglPanicEglImageIsDestroyed); |
|
60 |
++iCreateCount; |
|
61 |
} |
|
62 |
||
63 |
||
64 |
// If successful increments reference count & returns ETrue. (Ultimtately called from e.g. vgCreateEGLImageTargetKHR.) |
|
65 |
TBool CEglImage::OpenForVgImage(TSize& aSize, VGHandle& aVgHandle, TUint64& aSgImageId) |
|
66 |
{ |
|
67 |
EGL_TRACE("CEglImage::OpenForVgImage iCreateCount=%d, iOpenCount=%d", iCreateCount, iOpenCount); |
|
68 |
if ( (iCreateCount > 0) && (iSgImageInfo.iUsage & ESgUsageBitOpenVgImage) ) |
|
69 |
{ |
|
70 |
++iOpenCount; |
|
71 |
aSize = iSgImageInfo.iSizeInPixels; |
|
72 |
aVgHandle = iVgHandle; |
|
73 |
aSgImageId = iSgHandle.Id().iId; |
|
74 |
return ETrue; |
|
75 |
} |
|
76 |
return EFalse; |
|
77 |
} |
|
78 |
||
79 |
// If successful decrements reference count & returns ETrue. (Ultimtately called from e.g. vgDestroyImage.) |
|
80 |
TBool CEglImage::Close() |
|
81 |
{ |
|
82 |
EGL_TRACE("CEglImage::Close iCreateCount=%d, iOpenCount=%d", iCreateCount, iOpenCount); |
|
83 |
if (iOpenCount > 0) |
|
84 |
{ |
|
85 |
--iOpenCount; |
|
86 |
if ( RefCount() == 0 ) |
|
87 |
{ |
|
88 |
iSgHandle.Close(); |
|
89 |
} |
|
90 |
return ETrue; |
|
91 |
} |
|
92 |
return EFalse; |
|
93 |
} |
|
94 |
||
95 |
// If not already destroyed: mark EglImage as destroyed, dec reference count, return ETrue. (Called from eglDestroyImageKHR.) |
|
96 |
TBool CEglImage::Destroy() |
|
97 |
{ |
|
98 |
EGL_TRACE("CEglImage::Destroy iCreateCount=%d, iOpenCount=%d", iCreateCount, iOpenCount); |
|
99 |
if (!iIsDestroyed) |
|
100 |
{ |
|
101 |
--iCreateCount; |
|
102 |
if (iCreateCount == 0) |
|
103 |
{ |
|
104 |
iIsDestroyed = ETrue; |
|
105 |
if ( RefCount() == 0 ) |
|
106 |
{ |
|
107 |
iSgHandle.Close(); |
|
108 |
} |
|
109 |
return ETrue; |
|
110 |
} |
|
111 |
} |
|
112 |
return EFalse; |
|
113 |
} |
|
114 |
||
115 |
// Reference Count: count of unbalanced Create() plus Open() calls. |
|
116 |
TInt CEglImage::RefCount() |
|
117 |
{ |
|
118 |
return iOpenCount + iCreateCount; |
|
119 |
} |
|
120 |
||
121 |
TSgDrawableId CEglImage::SgImageId() const |
|
122 |
{ |
|
123 |
return iSgHandle.Id(); |
|
124 |
} |
|
125 |
||
126 |
EGLDisplay CEglImage::Display() const |
|
127 |
{ |
|
128 |
return iDisplay; |
|
129 |
} |
|
130 |
||
131 |
TBool CEglImage::IsDestroyed() const |
|
132 |
{ |
|
133 |
return iIsDestroyed; |
|
134 |
} |
|
135 |
#else |
|
136 |
void CEglImage::Duplicate() {} |
|
17
c9d1eb0bfac8
Get rid of pitch code because the host side does not need it
Faisal Memon <faisal.memon@nokia.com>
parents:
13
diff
changeset
|
137 |
TBool CEglImage::OpenForVgImage(TSize& aSize, VGHandle& aVgHandle, TUint64& aSgImageId) {return ETrue;} |
c9d1eb0bfac8
Get rid of pitch code because the host side does not need it
Faisal Memon <faisal.memon@nokia.com>
parents:
13
diff
changeset
|
138 |
TBool CEglImage::Close() {return ETrue;} |
c9d1eb0bfac8
Get rid of pitch code because the host side does not need it
Faisal Memon <faisal.memon@nokia.com>
parents:
13
diff
changeset
|
139 |
TBool CEglImage::Destroy() {return ETrue;} |
c9d1eb0bfac8
Get rid of pitch code because the host side does not need it
Faisal Memon <faisal.memon@nokia.com>
parents:
13
diff
changeset
|
140 |
TBool CEglImage::IsDestroyed() const {return ETrue;} |
c9d1eb0bfac8
Get rid of pitch code because the host side does not need it
Faisal Memon <faisal.memon@nokia.com>
parents:
13
diff
changeset
|
141 |
TInt CEglImage::RefCount() {return 0;} |
c9d1eb0bfac8
Get rid of pitch code because the host side does not need it
Faisal Memon <faisal.memon@nokia.com>
parents:
13
diff
changeset
|
142 |
EGLDisplay CEglImage::Display() const {return ETrue;} |
13 | 143 |
#endif |
144 |
||
145 |
||
146 |
//////////////////////////////////////////////////////////////////////////////////////////////////// |
|
147 |
// SgImage stuff in CGuestEGL |
|
148 |
//////////////////////////////////////////////////////////////////////////////////////////////////// |
|
149 |
||
150 |
||
151 |
EGLint CGuestEGL::ValidateEglImageTarget(EGLDisplay aDisplay, EGLContext aContext, EGLenum aTarget, |
|
152 |
EGLClientBuffer aBuffer, const EGLint *aAttribList, TSymbianPixmapTypeId aTargetPixmapType) |
|
153 |
{ |
|
154 |
if (!IsDisplayInitialized(aDisplay)) // ToDo just check display is valid, don't care if it is initialised |
|
155 |
{ |
|
156 |
return EGL_BAD_DISPLAY; |
|
157 |
} |
|
158 |
if ( (aContext != EGL_NO_CONTEXT) || (aTarget != EGL_NATIVE_PIXMAP_KHR) ) |
|
159 |
{ |
|
160 |
return EGL_BAD_PARAMETER; |
|
161 |
} |
|
162 |
aTargetPixmapType = EglInternalFunction_GetNativePixmapType((EGLNativePixmapType) aBuffer); |
|
163 |
if ( (aTargetPixmapType != EPixmapTypeSgImage) || !EglInternalFunction_IsValidNativePixmap((EGLNativePixmapType) aBuffer, aTargetPixmapType) ) |
|
164 |
{ |
|
165 |
return EGL_BAD_PARAMETER; |
|
166 |
} |
|
167 |
// ToDo check Attrib List is valid |
|
168 |
return EGL_SUCCESS; |
|
169 |
} |
|
170 |
||
171 |
#ifdef FAISALMEMON_S4_SGIMAGE |
|
172 |
EGLImageKHR CGuestEGL::DuplicateEglImageIfItExists(EGLDisplay aDisplay, TSgDrawableId aSgId) |
|
173 |
{ // ToDo something smarter with a Hash Map ? |
|
174 |
TInt imageCount = iEglImageArray.Count(); |
|
175 |
for (TInt index = 0; index < imageCount; ++index) |
|
176 |
{ |
|
177 |
CEglImage* imagePtr = iEglImageArray[index]; |
|
178 |
if ( !imagePtr->IsDestroyed() && (imagePtr->SgImageId() == aSgId) && (imagePtr->Display() == aDisplay) ) |
|
179 |
{ |
|
180 |
imagePtr->Duplicate(); |
|
181 |
EGL_TRACE("CGuestEGL::DuplicateEglImageIfItExists SgImage already used for eglImage=0x%x", imagePtr); |
|
182 |
return (EGLImageKHR)imagePtr; |
|
183 |
} |
|
184 |
} |
|
185 |
return EGL_NO_IMAGE_KHR; |
|
186 |
} |
|
187 |
||
188 |
||
189 |
||
190 |
EGLImageKHR CGuestEGL::CreateNewEglImage(TEglThreadState& aThreadState, EGLDisplay aDisplay, TSgDrawableId aSgId, TSgImageInfo aSgImageInfo) |
|
191 |
{ |
|
192 |
RHeap* oldHeap = CVghwUtils::SwitchToVghwHeap(); |
|
193 |
CEglImage* newImage = new CEglImage(); |
|
194 |
if (newImage) |
|
195 |
{ |
|
196 |
if (KErrNone != iEglImageArray.InsertInAddressOrder(newImage)) |
|
197 |
{ |
|
198 |
delete newImage; |
|
199 |
newImage = NULL; |
|
200 |
} |
|
201 |
} |
|
202 |
CVghwUtils::SwitchFromVghwHeap(oldHeap); |
|
203 |
if (newImage == NULL) |
|
204 |
{ |
|
205 |
EGL_TRACE("CGuestEGL::CreateNewEglImage 1.a error creating new CEglImage"); |
|
206 |
return EGL_NO_IMAGE_KHR; |
|
207 |
} |
|
208 |
||
209 |
// success ... |
|
210 |
EGL_TRACE("CGuestEGL::CreateNewEglImage 1.b Created new eglImage=0x%x", newImage); |
|
211 |
TInt pbufferHandle; |
|
212 |
TInt vgImageHandle; |
|
213 |
TUint64 sgHandles; |
|
214 |
EGLint hostResult = CVghwUtils::EglGetSgHandles(aSgId.iId, &sgHandles); |
|
215 |
pbufferHandle = (TInt)(sgHandles&0xFFFFFFFF); |
|
216 |
vgImageHandle = (TInt)((sgHandles>>32)&0xFFFFFFFF); |
|
217 |
EGL_TRACE("CGuestEGL::CreateNewEglImage 2. CVghwUtils::EglGetSgHandles result=%d, pbufferHandle=0x%x, vgImageHandle=0x%x", |
|
218 |
hostResult, pbufferHandle, vgImageHandle); |
|
219 |
EGLPANIC_ASSERT(KErrNone == hostResult, EEglPanicTemp); |
|
220 |
||
221 |
newImage->Create(aSgId, aDisplay, aSgImageInfo, pbufferHandle, vgImageHandle); |
|
222 |
||
223 |
return (EGLImageKHR)newImage; |
|
224 |
} |
|
225 |
||
226 |
#endif // FAISALMEMON_S4_SGIMAGE |
|
227 |
||
228 |
||
229 |
/* |
|
230 |
* eglCreateImageKHR supports Khronos EGL extension #8, "KHR_image_base" |
|
231 |
* |
|
232 |
* Supported values for target parameter: |
|
233 |
* EGL_NATIVE_PIXMAP_KHR for Khronos EGL extension #9, "KHR_image_pixmap" |
|
234 |
*/ |
|
235 |
EGLImageKHR CGuestEGL::eglCreateImageKHR(TEglThreadState& aThreadState, EGLDisplay aDisplay, EGLContext aContext, EGLenum aTarget, |
|
236 |
EGLClientBuffer aBuffer, const EGLint *aAttribList) |
|
237 |
{ |
|
238 |
// ToDo lock aDisplay |
|
239 |
EGLImageKHR eglImage = EGL_NO_IMAGE_KHR; |
|
240 |
iEglImageLock.Wait(); |
|
241 |
TSymbianPixmapTypeId targetPixmapType = EPixmapTypeNone; |
|
242 |
EGLint error = ValidateEglImageTarget(aDisplay, aContext, aTarget, aBuffer, aAttribList, targetPixmapType); |
|
243 |
EGL_TRACE("CGuestEGL::eglCreateImageKHR 1. validate error=%d, targetPixmapType=%d", error, targetPixmapType); |
|
244 |
||
245 |
if (error == EGL_SUCCESS) |
|
246 |
{ |
|
247 |
switch(targetPixmapType) |
|
248 |
{ |
|
249 |
case EPixmapTypeSgImage: |
|
250 |
{ |
|
251 |
#ifdef FAISALMEMON_S4_SGIMAGE |
|
252 |
// try to lock SgImage from disappearing by opening a temporary handle |
|
253 |
RSgImage* sgImagePtr = reinterpret_cast<RSgImage*>(aBuffer); |
|
254 |
TSgDrawableId sgId = sgImagePtr->Id(); |
|
255 |
RSgDrawable sgHandle; |
|
256 |
TSgImageInfo imgInfo; |
|
257 |
// ToDo ensure SgImage has correct suitable usage bits |
|
258 |
if ( (sgId != KSgNullDrawableId) && (KErrNone == sgHandle.Open(sgId)) && (KErrNone == sgImagePtr->GetInfo(imgInfo)) ) |
|
259 |
{ |
|
260 |
EGL_TRACE("CGuestEGL::eglCreateImageKHR 2. SgImage PixelFormat=%d; size=%d,%d; Usage=0x%x", |
|
261 |
imgInfo.iPixelFormat, imgInfo.iSizeInPixels.iWidth, imgInfo.iSizeInPixels.iHeight, imgInfo.iUsage); |
|
262 |
||
263 |
// find / create CEglImage |
|
264 |
if (error == EGL_SUCCESS) |
|
265 |
{ // SgImage may already be used for an EglImage |
|
266 |
eglImage = DuplicateEglImageIfItExists(aDisplay, sgId); |
|
267 |
if (eglImage == EGL_NO_IMAGE_KHR) |
|
268 |
{ // first use |
|
269 |
eglImage = CreateNewEglImage(aThreadState, aDisplay, sgId, imgInfo); |
|
270 |
if (eglImage == EGL_NO_IMAGE_KHR) |
|
271 |
{ |
|
272 |
error = EGL_BAD_ALLOC; |
|
273 |
} |
|
274 |
} |
|
275 |
} |
|
276 |
} |
|
277 |
else |
|
278 |
{ |
|
279 |
error = EGL_BAD_PARAMETER; |
|
280 |
} |
|
281 |
sgHandle.Close(); |
|
282 |
#endif |
|
283 |
} |
|
284 |
break; |
|
285 |
||
286 |
case EPixmapTypeFbsBitmap: |
|
287 |
case EPixmapTypeNone: |
|
288 |
default: // pixmap type not supported by this API |
|
289 |
error = EGL_BAD_PARAMETER; |
|
290 |
break; |
|
291 |
} |
|
292 |
} |
|
293 |
||
294 |
iEglImageLock.Signal(); |
|
295 |
aThreadState.SetEglError(error); |
|
296 |
||
297 |
// we either failed or should have a non-NULL handle |
|
298 |
EGLPANIC_ASSERT_DEBUG( (error != EGL_SUCCESS) || (eglImage != EGL_NO_IMAGE_KHR), EEglPanicTemp); |
|
299 |
||
300 |
return eglImage; |
|
301 |
} |
|
302 |
||
303 |
/* |
|
304 |
* eglDestroyImageKHR supports Khronos EGL extension #8, "KHR_image_base" |
|
305 |
*/ |
|
306 |
EGLBoolean CGuestEGL::eglDestroyImageKHR(TEglThreadState& aThreadState, EGLDisplay aDisplay, EGLImageKHR aImage) |
|
307 |
{ |
|
308 |
// ToDo lock aDisplay |
|
309 |
EGLBoolean result = EGL_FALSE; |
|
310 |
iEglImageLock.Wait(); |
|
311 |
TInt index = iEglImageArray.FindInAddressOrder((CEglImage*)aImage); |
|
312 |
if (index >= 0) |
|
313 |
{ |
|
314 |
CEglImage* eglImage = iEglImageArray[index]; |
|
315 |
if ( (eglImage->Display() && aDisplay) && eglImage->Destroy()) |
|
316 |
{ |
|
317 |
result = EGL_TRUE; |
|
318 |
if (eglImage->RefCount() == 0) |
|
319 |
{ |
|
320 |
iEglImageArray.Remove(index); |
|
321 |
delete eglImage; |
|
322 |
} |
|
323 |
} |
|
324 |
} |
|
325 |
iEglImageLock.Signal(); |
|
326 |
||
327 |
// set EGL Error appropriately |
|
328 |
aThreadState.SetEglError( result ? EGL_SUCCESS : EGL_BAD_PARAMETER); |
|
329 |
return result; |
|
330 |
} |
|
331 |
||
332 |
||
333 |
TBool CGuestEGL::EglImageOpenForVgImage(EGLImageKHR aImage, TSize& aSize, VGHandle& aVgHandle, TUint64& aSgImageId) |
|
334 |
{ |
|
335 |
EGLBoolean result = EGL_FALSE; |
|
336 |
iEglImageLock.Wait(); |
|
337 |
TInt index = iEglImageArray.FindInAddressOrder((CEglImage*)aImage); |
|
338 |
if (index >= 0) |
|
339 |
{ |
|
340 |
CEglImage* eglImage = iEglImageArray[index]; |
|
341 |
if (eglImage->OpenForVgImage(aSize, aVgHandle, aSgImageId)) |
|
342 |
{ |
|
343 |
result = EGL_TRUE; |
|
344 |
} |
|
345 |
} |
|
346 |
iEglImageLock.Signal(); |
|
347 |
return result; |
|
348 |
} |
|
349 |
||
350 |
void CGuestEGL::EglImageClose(EGLImageKHR aImage) |
|
351 |
{ |
|
352 |
iEglImageLock.Wait(); |
|
353 |
TInt index = iEglImageArray.FindInAddressOrder((CEglImage*)aImage); |
|
354 |
if (index >= 0) |
|
355 |
{ |
|
356 |
CEglImage* eglImage = iEglImageArray[index]; |
|
357 |
if (eglImage->Close()) |
|
358 |
{ |
|
359 |
if (eglImage->RefCount() == 0) |
|
360 |
{ |
|
361 |
iEglImageArray.Remove(index); |
|
362 |
delete eglImage; |
|
363 |
} |
|
364 |
} |
|
365 |
} |
|
366 |
iEglImageLock.Signal(); |
|
367 |
} |
|
368 |
||
369 |
// end of file eglsgimage.cpp |