os/graphics/graphicsresourceservices/graphicsresourceimplementation/src/sgdriver.cpp
Update contrib.
1 // Copyright (c) 2007-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 // Graphics Resource - driver implementation
19 const TInt KSgMaxLocalChunkSize = 0x100000;
20 const TInt KSgLocalChunkAlignment = 0x40;
22 EXPORT_C TInt RSgDriver::Open()
28 if (gPls.iError != KErrNone)
41 TInt err = chunk.CreateLocal(KMinHeapGrowBy, KSgMaxLocalChunkSize, EOwnerProcess);
47 RHeap* heap = UserHeap::ChunkHeap(chunk, 0, KMinHeapGrowBy, KSgMaxLocalChunkSize, KSgLocalChunkAlignment, ETrue);
48 __ASSERT_DEBUG(heap, Panic(ESgPanicResourceImplGeneral));
49 XSgDriver* driver = static_cast<XSgDriver*>(heap->Alloc(sizeof(XSgDriver)));
50 __ASSERT_DEBUG(driver, Panic(ESgPanicResourceImplGeneral));
51 new(driver) XSgDriver(heap);
52 err = driver->Construct();
60 iImpl = gPls.iDriver = driver;
65 EXPORT_C void RSgDriver::Close()
69 __ASSERT_DEBUG(gPls.iError == KErrNone, Panic(ESgPanicResourceImplGeneral));
71 __ASSERT_DEBUG(gPls.iOpenCount > 0, Panic(ESgPanicResourceImplGeneral));
72 if (--gPls.iOpenCount == 0)
74 gPls.iDriver->Delete();
82 EXPORT_C TInt RSgDriver::GetInterface(TUid aInterfaceUid, TAny*& aInterfacePtr) const
89 return static_cast<XSgDriver*>(iImpl)->GetInterface(aInterfaceUid, aInterfacePtr);
92 EXPORT_C TVersion RSgDriver::Version()
94 return TVersion(1, 1, 1);
97 _LIT(KSgResourceImplPanicCategory, "SGRES-IMPL");
99 void Panic(TSgResourceImplPanicReason aReason)
101 User::Panic(KSgResourceImplPanicCategory, aReason);
104 #ifdef SYMBIAN_GRAPHICS_USE_GPU
105 const TUint32 KSgResourceAttributes = 0;
107 const TUint32 KSgResourceAttributes = ESgResourceAttrCpuCached;
110 LOCAL_C TInt SgMinDataStride(TInt aWidthInPixels, TInt aPixelFormat)
112 switch (aPixelFormat)
114 case ESgPixelFormatA_8:
115 return aWidthInPixels;
116 case ESgPixelFormatRGB_565:
117 return aWidthInPixels << 1;
118 case ESgPixelFormatXRGB_8888:
119 case ESgPixelFormatARGB_8888:
120 case ESgPixelFormatARGB_8888_PRE:
121 return aWidthInPixels << 2;
124 Panic(ESgPanicResourceImplGeneral);
130 LOCAL_C TInt SgAlignedDataStride(TInt aWidthInPixels, TInt aPixelFormat)
132 #if defined(SYMBIAN_GRAPHICS_USE_MBX)
133 // MBX requires 2^n stride
134 for (TInt width = 8; width & KMaxTInt; width <<= 1)
136 if (width >= aWidthInPixels)
138 aWidthInPixels = width;
142 #elif defined(SYMBIAN_GRAPHICS_USE_SGX)
143 // SGX requires 32-pixel alignment
144 aWidthInPixels = (aWidthInPixels + 31) & ~31;
146 return Align4(SgMinDataStride(aWidthInPixels, aPixelFormat));
149 XSgDriverPls::XSgDriverPls()
151 iError = iMutex.CreateLocal();
156 XSgDriver::XSgDriver(RHeap* aHeap)
157 : iHeap(aHeap), iHasOpenVg(EFalse), iHasOpenGles(EFalse), iHasOpenGles2(EFalse)
161 XSgDriver::~XSgDriver()
165 __ASSERT_DEBUG(iImagesByAddress.Count() == 0, Panic(ESgPanicUnclosedResources));
166 __ASSERT_DEBUG(iImagesById.Count() == 0, Panic(ESgPanicUnclosedResources));
169 TInt XSgDriver::Construct()
171 TInt err = iMutex.CreateLocal();
176 err = User::LoadLogicalDevice(KSgDeviceName);
177 if (err != KErrNone && err != KErrAlreadyExists)
181 err = iDevice.Connect();
186 _LIT(KLibOpenVg, "libOpenVG.dll");
187 _LIT(KLibOpenGles, "libGLESv1_CM.dll");
188 _LIT(KLibOpenGles2, "libGLESv2.dll");
190 if (lib.Load(KLibOpenVg) == KErrNone)
195 if (lib.Load(KLibOpenGles) == KErrNone)
198 iHasOpenGles = ETrue;
200 if (lib.Load(KLibOpenGles2) == KErrNone)
203 iHasOpenGles2 = ETrue;
208 void XSgDriver::Delete()
213 __ASSERT_DEBUG(heap->Count() == 0, Panic(ESgPanicUnclosedResources));
217 TInt XSgDriver::CreateImage(const TSgImageInfo& aInfo, const TAny* aDataAddress, TInt aDataStride, const TSgAttributeArrayBase* aAttributes, TAny*& aResult)
219 TInt err = CheckImageInfo(aInfo);
224 TInt minDataStride = SgMinDataStride(aInfo.iSizeInPixels.iWidth, aInfo.iPixelFormat);
225 if (aDataAddress && Abs(aDataStride) < minDataStride)
231 return KErrNotSupported;
233 TUint32 attribs = KSgResourceAttributes | MatchingEglConfigUsage(aInfo.iUsage);
234 TPckgBuf<TSgImageMetaData> metaDataPckg;
235 metaDataPckg().iSizeInPixels = aInfo.iSizeInPixels;
236 metaDataPckg().iPixelFormat = aInfo.iPixelFormat;
237 TInt dataStride = SgAlignedDataStride(aInfo.iSizeInPixels.iWidth, aInfo.iPixelFormat);
238 TInt dataSize = dataStride * aInfo.iSizeInPixels.iHeight;
240 err = iDevice.CreateResource(attribs, metaDataPckg, dataSize, id.iId);
246 XSgImage* imageImpl = static_cast<XSgImage*>(iHeap->Alloc(sizeof(XSgImage)));
249 (void)iDevice.CloseResource(id.iId);
253 new(imageImpl) XSgImage(id, attribs, metaDataPckg(), iDevice.ResourceDataAddress(id.iId), dataStride);
254 RHeap* prevHeap = User::SwitchHeap(iHeap);
255 err = iImagesByAddress.InsertInAddressOrder(imageImpl);
258 err = iImagesById.InsertInOrder(imageImpl, XSgImage::Compare);
261 iImagesByAddress.Remove(iImagesByAddress.FindInAddressOrder(imageImpl));
262 iImagesByAddress.GranularCompress();
265 User::SwitchHeap(prevHeap);
270 const TAny* src = aDataStride > 0 ? aDataAddress : PtrAdd(aDataAddress, -aDataStride * (aInfo.iSizeInPixels.iHeight - 1));
271 TAny* trg = imageImpl->DataAddress();
272 for (TInt y = 0; y < aInfo.iSizeInPixels.iHeight; ++y)
274 Mem::Copy(trg, src, minDataStride);
275 src = PtrAdd(src, aDataStride);
276 trg = PtrAdd(trg, dataStride);
283 (void)iDevice.CloseResource(id.iId);
284 iHeap->Free(imageImpl);
290 TInt XSgDriver::CreateImage(const TSgImageInfo& aInfo, const XSgImage* aImageImpl, const TSgAttributeArrayBase* aAttributes, TAny*& aResult)
296 __ASSERT_ALWAYS(CheckImage(aImageImpl), Panic(ESgPanicBadDrawableHandle));
298 aImageImpl->GetInfo(info);
299 if (aInfo.iSizeInPixels != info.iSizeInPixels || aInfo.iPixelFormat != info.iPixelFormat)
301 return KErrNotSupported;
303 return CreateImage(aInfo, aImageImpl->DataAddress(), aImageImpl->DataStride(), aAttributes, aResult);
306 TInt XSgDriver::FindAndOpenImage(TSgDrawableId aId, const TSgAttributeArrayBase* aAttributes, TAny*& aResult)
308 if (aId == KSgNullDrawableId)
314 return KErrNotSupported;
318 TInt indexById = iImagesById.FindInOrder(aId, XSgImage::Compare);
321 XSgImage* imageImpl = iImagesById[indexById];
322 err = imageImpl->Open();
330 err = iDevice.OpenResource(aId.iId);
336 TPckgBuf<TSgImageMetaData> metaDataPckg;
337 err = iDevice.GetResourceMetaData(aId.iId, metaDataPckg);
338 if (metaDataPckg.Size() != sizeof(TSgImageMetaData))
345 attribs = iDevice.ResourceAttributes(aId.iId);
346 TSgImageInfo info(metaDataPckg().iSizeInPixels, metaDataPckg().iPixelFormat, attribs & KSgUsageBitMask);
347 if (CheckImageInfo(info) != KErrNone)
355 dataStride = SgAlignedDataStride(metaDataPckg().iSizeInPixels.iWidth, metaDataPckg().iPixelFormat);
356 if (iDevice.ResourceDataSize(aId.iId) != dataStride * metaDataPckg().iSizeInPixels.iHeight)
363 (void)iDevice.CloseResource(aId.iId);
367 XSgImage* imageImpl = static_cast<XSgImage*>(iHeap->Alloc(sizeof(XSgImage)));
370 (void)iDevice.CloseResource(aId.iId);
374 new(imageImpl) XSgImage(aId, attribs, metaDataPckg(), iDevice.ResourceDataAddress(aId.iId), dataStride);
375 RHeap* prevHeap = User::SwitchHeap(iHeap);
376 err = iImagesByAddress.InsertInAddressOrder(imageImpl);
379 err = iImagesById.InsertInOrder(imageImpl, XSgImage::Compare);
382 iImagesByAddress.Remove(iImagesByAddress.FindInAddressOrder(imageImpl));
383 iImagesByAddress.GranularCompress();
386 User::SwitchHeap(prevHeap);
393 (void)iDevice.CloseResource(aId.iId);
394 iHeap->Free(imageImpl);
401 void XSgDriver::DeleteImage(XSgImage* aImageImpl)
404 TInt indexByAddress = iImagesByAddress.FindInAddressOrder(aImageImpl);
405 TInt indexById = iImagesById.FindInOrder(aImageImpl, XSgImage::Compare);
406 __ASSERT_DEBUG(indexByAddress >= 0 && indexById >= 0, Panic(ESgPanicBadImagePointer));
407 RHeap* prevHeap = User::SwitchHeap(iHeap);
408 iImagesByAddress.Remove(indexByAddress);
409 iImagesById.Remove(indexById);
410 iImagesByAddress.GranularCompress();
411 iImagesById.GranularCompress();
412 User::SwitchHeap(prevHeap);
413 (void)iDevice.CloseResource(aImageImpl->Id().iId);
414 aImageImpl->~XSgImage();
415 iHeap->Free(aImageImpl);
419 TUint32 XSgDriver::MatchingEglConfigUsage(TUint32 aUsage) const
421 if (aUsage & KSgUsageAllSurfaceTypes)
425 aUsage |= ESgUsageBitOpenVgSurface;
429 aUsage |= ESgUsageBitOpenGlesSurface;
433 aUsage |= ESgUsageBitOpenGles2Surface;
439 TInt XSgDriver::CheckImageInfo(const TSgImageInfo& aInfo) const
441 if (aInfo.iSizeInPixels.iWidth <= 0 || aInfo.iSizeInPixels.iHeight <= 0
442 || aInfo.iPixelFormat == EUidPixelFormatUnknown || aInfo.iUsage == 0)
446 if (aInfo.iSizeInPixels.iWidth > KMaxTInt16 / 2 || aInfo.iSizeInPixels.iHeight > KMaxTInt16 / 2)
450 if (aInfo.iUsage & ~KSgUsageAll)
452 return KErrNotSupported;
454 if ((aInfo.iUsage & (ESgUsageBitOpenVgImage | ESgUsageBitOpenVgSurface)) && !iHasOpenVg
455 || (aInfo.iUsage & (ESgUsageBitOpenGlesTexture2D | ESgUsageBitOpenGlesSurface)) && !iHasOpenGles
456 || (aInfo.iUsage & (ESgUsageBitOpenGles2Texture2D | ESgUsageBitOpenGles2Surface)) && !iHasOpenGles2)
458 return KErrNotSupported;
460 switch (aInfo.iPixelFormat)
462 case ESgPixelFormatA_8:
463 if (aInfo.iUsage & KSgUsageAllSurfaceTypes)
465 return KErrNotSupported;
467 // coverity[fallthrough]
468 case ESgPixelFormatRGB_565:
469 case ESgPixelFormatXRGB_8888:
470 case ESgPixelFormatARGB_8888:
471 case ESgPixelFormatARGB_8888_PRE:
474 return KErrNotSupported;
478 TBool XSgDriver::CheckImage(const TAny* aImageImpl) const
481 TInt indexByAddress = iImagesByAddress.FindInAddressOrder(static_cast<const XSgImage*>(aImageImpl));
483 return indexByAddress >= 0;
486 TInt XSgDriver::GetInterface(TUid aInterfaceUid, TAny*& aInterfacePtr)
488 if (aInterfaceUid == KNullUid)
492 if (aInterfaceUid.iUid == MSgDriver_Profiling::EInterfaceUid)
494 aInterfacePtr = static_cast<MSgDriver_Profiling*>(this);
498 if (aInterfaceUid.iUid == MSgDriver_Test::EInterfaceUid)
500 aInterfacePtr = static_cast<MSgDriver_Test*>(this);
504 return KErrExtensionNotSupported;
507 TInt XSgDriver::LocalResourceCount() const
511 TInt n = iImagesByAddress.Count();
512 for (TInt i = 0; i < n; ++i)
514 count += iImagesByAddress[i]->RefCount();
520 TInt XSgDriver::GlobalResourceCount() const
522 return iDevice.GlobalResourceCount();
525 TInt XSgDriver::LocalGraphicsMemoryUsed() const
527 return iDevice.LocalGraphicsMemoryUsed();
530 TInt XSgDriver::GlobalGraphicsMemoryUsed() const
532 return iDevice.GlobalGraphicsMemoryUsed();
537 void XSgDriver::AllocMarkStart()
540 iHeap->__DbgMarkStart();
544 void XSgDriver::AllocMarkEnd(TInt aCount)
547 TUint32 badCell = iHeap->__DbgMarkEnd(aCount);
551 _LIT(KPanicCategoryFormat, "SGALLOC:%08x");
553 category.Format(KPanicCategoryFormat, badCell);
554 User::Panic(category, 0);
558 void XSgDriver::SetAllocFail(RAllocator::TAllocFail aType, TInt aRate)
561 iHeap->__DbgSetAllocFail(aType, aRate);