os/graphics/graphicsresourceservices/graphicsresourceadaptation/src/sgdriverimpl.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.
17 #include "sgdriverimpl.h"
18 #include "sgimageimpl.h"
19 #include "sgimagecollectionimpl.h"
22 TInt SgAlignedDataStride(TInt aWidthInPixels, TUidPixelFormat aPixelFormat)
24 #if defined(SYMBIAN_GRAPHICS_USE_MBX)
25 // MBX requires 2^n stride.
26 for (TInt width = 8; width & KMaxTInt; width <<= 1)
28 if (width >= aWidthInPixels)
30 aWidthInPixels = width;
34 #elif defined(SYMBIAN_GRAPHICS_USE_SGX)
35 // SGX requires a 32 pixel alignment.
36 aWidthInPixels = (aWidthInPixels + 31) & ~31;
38 return Align4(SgMinDataStride(aWidthInPixels, aPixelFormat));
42 TInt SgOffsetToFirstBuffer(TInt aMetaDataSize)
44 #if defined(SYMBIAN_GRAPHICS_USE_MBX) || defined(SYMBIAN_GRAPHICS_USE_SGX)
45 return (aMetaDataSize + 127) & ~127;
47 return Align4(aMetaDataSize);
52 TInt SgOffsetBetweenBuffers(TInt aDataStride, TInt aScanLineCount)
54 #if defined(SYMBIAN_GRAPHICS_USE_MBX)
55 for (TInt count = 16; count & KMaxTInt; count <<= 1)
57 if (count >= aScanLineCount)
59 aScanLineCount = count;
64 return aDataStride * aScanLineCount;
68 TBool SgIsValidImageInfo(const TSgImageInfo& aInfo)
70 return aInfo.iSizeInPixels.iWidth > 0 && aInfo.iSizeInPixels.iHeight > 0
71 && (aInfo.iUsage & ESgUsageScreenSource ? aInfo.iScreenId >= KSgScreenIdMain : aInfo.iScreenId >= KSgScreenIdAny)
72 && (aInfo.iUserAttributes ? aInfo.iUserAttributeCount > 0 : aInfo.iUserAttributeCount == 0);
76 TBool SgIsMutableImage(const TSgImageInfo& aInfo)
78 return (aInfo.iCpuAccess & ESgCpuAccessWritable) || (aInfo.iUsage & KSgUsageAllTargets);
82 TBool SgIsCachedImage(const TSgImageInfo& aInfo)
84 #ifdef SYMBIAN_GRAPHICS_USE_GPU
85 #ifdef SYMBIAN_GRAPHICS_USE_CACHE
86 return aInfo.iCpuAccess != ESgCpuAccessNone;
98 // TSgPixelFormatTableEntry
100 TBool TSgPixelFormatTableEntry::IsMatch(const TSgImageInfo& aInfo) const
102 return iPixelFormat == aInfo.iPixelFormat
103 && !(~iUsage & aInfo.iUsage)
104 && !(~iCpuAccess & aInfo.iCpuAccess)
105 && (iScreenId == KSgScreenIdAny || iScreenId == aInfo.iScreenId);
109 TBool TSgPixelFormatTableEntry::IsMatchIgnoringPixelFormat(const TSgImageInfo& aInfo) const
111 return !(~iUsage & aInfo.iUsage)
112 && !(~iCpuAccess & aInfo.iCpuAccess)
113 && (iScreenId == KSgScreenIdAny || iScreenId == aInfo.iScreenId);
117 TBool TSgPixelFormatTableEntry::IsMatchIgnoringUsage(const TSgImageInfo& aInfo) const
119 return iPixelFormat == aInfo.iPixelFormat
120 && !(~iCpuAccess & aInfo.iCpuAccess)
121 && (iScreenId == KSgScreenIdAny || iScreenId == aInfo.iScreenId);
129 __ASSERT_DEBUG(iDriverImpl.IsMutexHeld(), Panic(ESgPanicMutexNotHeld));
130 __ASSERT_DEBUG(iRefCount == 0, Panic(ESgPanicBadReferenceCount));
134 void XSgBase::Delete()
136 XSgDriverImpl& driverImpl = iDriverImpl;
138 driverImpl.Free(this);
144 TInt XSgDriverImpl::Construct()
146 TInt err = iMutex.CreateLocal();
151 err = iSurfaceManager.Open();
156 return ConstructPixelFormatTable();
160 XSgDriverImpl::~XSgDriverImpl()
162 __ASSERT_DEBUG(iImages.Count() == 0, Panic(ESgPanicUnclosedResources));
163 __ASSERT_DEBUG(iImageCollections.Count() == 0, Panic(ESgPanicUnclosedResources));
165 iSurfaceManager.Close();
166 RHeap* prevHeap = User::SwitchHeap(iHeap);
167 iPixelFormatTable.Close();
168 iLastPixelFormats.Close();
169 User::SwitchHeap(prevHeap);
173 void XSgDriverImpl::Delete()
176 this->~XSgDriverImpl();
178 __ASSERT_DEBUG(heap->Count() == 0, Panic(ESgPanicUnclosedResources));
183 TInt XSgDriverImpl::MapSurface(const TSurfaceId& aSurfaceId, RChunk& aChunk)
186 TInt err = iSurfaceManager.MapSurface(aSurfaceId, chunk);
192 err = aChunk.Duplicate(RThread()); // Get a process-wide handle
198 TArray<TSgPixelFormatTableEntry> XSgDriverImpl::PixelFormatTable() const
200 return iPixelFormatTable.Array();
204 TInt XSgDriverImpl::CanCreateImage(const TSgImageInfo& aInfo) const
206 __ASSERT_DEBUG(SgIsValidImageInfo(aInfo), Panic(ESgPanicBadImageInfo));
207 if (aInfo.iSizeInPixels.iWidth > KMaxTInt16 / 2 || aInfo.iSizeInPixels.iHeight > KMaxTInt16 / 2)
211 TArray<TSgPixelFormatTableEntry> pixelFormatTable = PixelFormatTable();
212 TInt n = pixelFormatTable.Count();
213 for (TInt i = 0; i < n; ++i)
215 if (pixelFormatTable[i].IsMatch(aInfo))
217 return SgMinDataStride(aInfo.iSizeInPixels.iWidth, aInfo.iPixelFormat) > 0 ? KErrNone : KErrNotSupported;
220 return KErrNotSupported;
224 TInt XSgDriverImpl::GetPixelFormats(const TSgImageInfo& aInfo, TUidPixelFormat* aPixelFormats, TInt& aCount)
226 if (!SgIsValidImageInfo(aInfo) || (aPixelFormats && aCount < 0))
231 if (aInfo.iSizeInPixels != iLastSizeInPixels
232 || aInfo.iUsage != iLastUsage
233 || aInfo.iCpuAccess != iLastCpuAccess
234 || aInfo.iScreenId != iLastScreenId)
236 iLastSizeInPixels = aInfo.iSizeInPixels;
237 iLastUsage = aInfo.iUsage;
238 iLastCpuAccess = aInfo.iCpuAccess;
239 iLastScreenId = aInfo.iScreenId;
240 RHeap* prevHeap = User::SwitchHeap(iHeap);
241 iLastPixelFormats.Reset();
242 User::SwitchHeap(prevHeap);
243 TArray<TSgPixelFormatTableEntry> pixelFormatTable = PixelFormatTable();
244 TInt n = pixelFormatTable.Count();
245 for (TInt i = 0; i < n; ++i)
247 const TSgPixelFormatTableEntry& entry = pixelFormatTable[i];
248 if (entry.IsMatchIgnoringPixelFormat(aInfo) && iLastPixelFormats.Find(entry.iPixelFormat) == KErrNotFound)
250 User::SwitchHeap(iHeap);
251 TInt err = iLastPixelFormats.Append(entry.iPixelFormat);
252 User::SwitchHeap(prevHeap);
255 iLastSizeInPixels = TSize(0, 0);
256 iLastUsage = ESgUsageNone;
257 iLastCpuAccess = ESgCpuAccessNone;
258 iLastScreenId = KSgScreenIdMain;
259 User::SwitchHeap(iHeap);
260 iLastPixelFormats.Reset();
261 User::SwitchHeap(prevHeap);
271 TInt n = Min(aCount, iLastPixelFormats.Count());
272 for (TInt i = 0; i < n; ++i)
274 aPixelFormats[i] = iLastPixelFormats[i];
276 if (aCount < iLastPixelFormats.Count())
281 aCount = iLastPixelFormats.Count();
287 TInt XSgDriverImpl::CreateImage(const TSgImageInfo& aInfo, const TAny* aDataAddress, TInt aDataStride, MSgDrawableAdapter*& aResult)
289 if (!SgIsValidImageInfo(aInfo))
297 if (!aDataAddress && !SgIsMutableImage(aInfo))
299 return KErrNoInitializationData;
301 TInt err = CanCreateImage(aInfo);
306 #if !defined(SYMBIAN_GRAPHICS_USE_GPU) && !defined(__WINS__)
307 if (!aInfo.iShareable)
313 TSgImageId_SwLocal id_SwLocal;
317 id_SwLocal.iProcessId = RProcess().Id();
318 id_SwLocal.iRandom[0] = Math::Random();
319 id_SwLocal.iRandom[1] = Math::Random();
320 id_SwLocal.iMinusOne = KErrNotFound;
321 id_SwLocal.iFlags = 0;
323 while (iImages.FindInOrder(id, XSgImageImplBase::Compare) != KErrNotFound);
324 XSgImageImpl_SwLocal* impl;
325 err = XSgImageImpl_SwLocal::New(impl, *this, id, aInfo, aDataAddress, aDataStride);
331 RHeap* prevHeap = User::SwitchHeap(iHeap);
332 err = iImages.InsertInOrder(impl, XSgImageImplBase::Compare);
333 User::SwitchHeap(prevHeap);
347 XSgImageImpl_SurfaceManager* impl;
348 err = XSgImageImpl_SurfaceManager::New(impl, *this, aInfo, SgIsCachedImage(aInfo), aDataAddress, aDataStride);
354 RHeap* prevHeap = User::SwitchHeap(iHeap);
355 err = iImages.InsertInOrder(impl, XSgImageImplBase::Compare);
356 User::SwitchHeap(prevHeap);
370 TInt XSgDriverImpl::CreateImage(const TSgImageInfo& aInfo, MSgImageAdapter* aImage, MSgDrawableAdapter*& aResult)
376 __ASSERT_DEBUG(CheckImage(*aImage), Panic(ESgPanicBadImageHandle));
377 XSgImageImplBase& impl = static_cast<XSgImageImplBase&>(*aImage);
378 if (aInfo.iSizeInPixels != impl.MetaData().iSizeInPixels
379 || aInfo.iPixelFormat != impl.MetaData().iPixelFormat)
381 return KErrNotSupported;
383 TInt err = impl.BeginDataAccess(ESgCpuAccessReadOnly);
388 err = CreateImage(aInfo, impl.DataAddress(), impl.DataStride(), aResult);
389 impl.EndDataAccess();
394 TInt XSgDriverImpl::CreateImageCollection(const TSgImageInfo& aInfo, TInt aImageCount, MSgImageCollectionAdapter*& aResult)
396 if (!SgIsValidImageInfo(aInfo) || aImageCount <= 0)
404 if (!SgIsMutableImage(aInfo))
406 return KErrNotSupported;
408 TInt err = CanCreateImage(aInfo);
414 XSgImageCollectionImpl* impl;
415 const TInt stride = SgAlignedDataStride(aInfo.iSizeInPixels.iWidth, aInfo.iPixelFormat);
416 const TInt offsetToFirst = SgOffsetToFirstBuffer(sizeof(TSgImageMetaData));
417 const TInt offsetBetween = SgOffsetBetweenBuffers(stride, aInfo.iSizeInPixels.iHeight);
419 err = XSgImageCollectionImpl::New(impl, *this, aInfo, aImageCount, SgIsCachedImage(aInfo), stride, offsetToFirst, offsetBetween);
425 RHeap* prevHeap = User::SwitchHeap(iHeap);
426 err = iImageCollections.InsertInAddressOrder(impl);
427 User::SwitchHeap(prevHeap);
440 TInt XSgDriverImpl::GetBufferOffset(const TSurfaceId& aSurfaceID, TInt aBuffer, TInt &aOffset)
442 return iSurfaceManager.GetBufferOffset(aSurfaceID,aBuffer,aOffset);
445 TInt XSgDriverImpl::GetSurfaceManagerAttrib(RSurfaceManager::TSurfaceManagerAttrib aAttrib, TInt& aValue)
447 return iSurfaceManager.GetSurfaceManagerAttrib(aAttrib,aValue);
450 TInt XSgDriverImpl::CreateImageCollections(const TSgImageInfo aInfos[], TInt aImageCount,
451 MSgImageCollectionAdapter* aCollections[], TInt aCollectionCount)
453 if (aImageCount <= 0 || aCollectionCount <= 0)
457 TBool isCached = EFalse;
458 TInt offsetToFirstBuffer = SgOffsetToFirstBuffer(aCollectionCount * sizeof(TSgImageMetaData));
459 TInt maxOffsetBetweenBuffers = 0;
460 for (TInt i = 0; i < aCollectionCount; ++i)
462 const TSgImageInfo& info = aInfos[i];
463 if (!SgIsValidImageInfo(info))
471 if (!SgIsMutableImage(info))
473 return KErrNotSupported;
475 TInt err = CanCreateImage(info);
480 if (SgIsCachedImage(info))
484 TInt stride = SgAlignedDataStride(info.iSizeInPixels.iWidth, info.iPixelFormat);
485 TInt offsetBetweenBuffers = SgOffsetBetweenBuffers(stride, info.iSizeInPixels.iHeight);
486 if (offsetBetweenBuffers > maxOffsetBetweenBuffers)
488 maxOffsetBetweenBuffers = offsetBetweenBuffers;
492 XSgImageCollectionImpl* firstImpl = NULL;
493 for (TInt i = 0; i < aCollectionCount; ++i)
495 const TInt stride = SgAlignedDataStride(aInfos[i].iSizeInPixels.iWidth, aInfos[i].iPixelFormat);
496 XSgImageCollectionImpl* impl;
497 TInt err = XSgImageCollectionImpl::New(impl, *this, aInfos[i], aImageCount, isCached,
498 stride, offsetToFirstBuffer, maxOffsetBetweenBuffers, i, firstImpl);
504 RSurfaceManager::TInfoBuf info;
505 TSurfaceId surface=impl->SurfaceId();
506 err = SurfaceInfo(surface, info);
509 // get the actual value used for offset to first buffer
510 err = iSurfaceManager.GetBufferOffset(surface, 0, offsetToFirstBuffer);
511 // get the actual value used for offset between buffers
514 TInt offsetToSecondBuffer;
515 err = iSurfaceManager.GetBufferOffset(surface, 1, offsetToSecondBuffer);
516 maxOffsetBetweenBuffers = offsetToSecondBuffer - offsetToFirstBuffer;
520 maxOffsetBetweenBuffers = 0;
530 RHeap* prevHeap = User::SwitchHeap(iHeap);
531 err = iImageCollections.InsertInAddressOrder(impl);
532 User::SwitchHeap(prevHeap);
536 aCollections[i] = impl;
547 aCollections[i]->Close();
548 aCollections[i] = NULL;
559 TInt XSgDriverImpl::OpenDrawable(const TSgDrawableId& aId, TUint32 aMode, TUid aHandleType, MSgDrawableAdapter*& aResult)
561 if (aHandleType == KSgImageTypeUid || aHandleType == KSgDrawableTypeUid)
563 return OpenImage(aId, aMode, aResult);
565 return KErrNotSupported;
569 TInt XSgDriverImpl::OpenImage(const TSgDrawableId& aId, TUint32 aMode, MSgDrawableAdapter*& aResult)
576 TSgDrawableId id = aId;
577 id.iId[KSgImageIdFlagsIndex] |= aMode;
578 TInt i = iImages.FindInOrder(id, XSgImageImplBase::CompareIgnoringFlags);
579 if (i != KErrNotFound)
581 XSgImageImplBase* impl;
582 TInt j = iImages.FindInOrder(id, XSgImageImplBase::Compare);
583 if (j != KErrNotFound)
590 #ifndef SYMBIAN_GRAPHICS_USE_GPU
591 if (TSgImageId_SwLocal::IsMatch(id))
593 XSgImageImpl_SwLocal* impl2;
594 TInt err = XSgImageImpl_SwLocal::New(impl2, *static_cast<XSgImageImpl_SwLocal*>(impl), id.iId[KSgImageIdFlagsIndex]);
604 if (TSgImageId_SurfaceManager::IsMatch(id))
606 XSgImageImpl_SurfaceManager* impl2;
607 TInt err = XSgImageImpl_SurfaceManager::New(impl2, *static_cast<XSgImageImpl_SurfaceManager*>(impl), id.iId[KSgImageIdFlagsIndex]);
620 RHeap* prevHeap = User::SwitchHeap(iHeap);
621 TInt err = iImages.InsertInOrder(impl, XSgImageImplBase::Compare);
622 User::SwitchHeap(prevHeap);
635 TInt err = KErrNotFound;
636 if (TSgImageId_SurfaceManager::IsMatch(id))
638 XSgImageImpl_SurfaceManager* impl;
639 err = XSgImageImpl_SurfaceManager::New(impl, *this, id);
645 if (!impl->MetaData().iShareable && impl->MetaData().iCreatorProcess != RProcess().Id())
649 return KErrPermissionDenied;
651 RHeap* prevHeap = User::SwitchHeap(iHeap);
652 err = iImages.InsertInOrder(impl, XSgImageImplBase::Compare);
653 User::SwitchHeap(prevHeap);
668 void XSgDriverImpl::DeleteImage(XSgImageImplBase* aImage)
670 __ASSERT_DEBUG(iMutex.IsHeld(), Panic(ESgPanicMutexNotHeld));
671 TInt i = iImages.FindInOrder(aImage, XSgImageImplBase::Compare);
672 __ASSERT_DEBUG(i != KErrNotFound, Panic(ESgPanicBadImageHandle));
673 RHeap* prevHeap = User::SwitchHeap(iHeap);
675 iImages.GranularCompress();
676 User::SwitchHeap(prevHeap);
681 void XSgDriverImpl::DeleteImageCollection(XSgImageCollectionImpl* aImageCollection)
683 __ASSERT_DEBUG(iMutex.IsHeld(), Panic(ESgPanicMutexNotHeld));
684 TInt i = iImageCollections.FindInAddressOrder(aImageCollection);
685 __ASSERT_DEBUG(i != KErrNotFound, Panic(ESgPanicBadImageCollectionHandle));
686 RHeap* prevHeap = User::SwitchHeap(iHeap);
687 iImageCollections.Remove(i);
688 iImageCollections.GranularCompress();
689 User::SwitchHeap(prevHeap);
690 aImageCollection->Delete();
694 TBool XSgDriverImpl::CheckDrawable(const MSgResourceAdapter& aDrawable) const
697 TInt i = iImages.Find(&static_cast<const XSgImageImplBase&>(aDrawable));
699 return i != KErrNotFound;
703 TBool XSgDriverImpl::CheckImage(const MSgResourceAdapter& aImage) const
706 TInt i = iImages.Find(&static_cast<const XSgImageImplBase&>(aImage));
708 return i != KErrNotFound;
712 TBool XSgDriverImpl::CheckImageCollection(const MSgResourceAdapter& aImageCollection) const
715 TInt i = iImageCollections.Find(&static_cast<const XSgImageCollectionImpl&>(aImageCollection));
717 return i != KErrNotFound;
721 TInt XSgDriverImpl::ResourceCount() const
725 for (TInt i = 0; i < iImages.Count(); ++i)
727 count += iImages[i]->RefCount();
729 for (TInt i = 0; i < iImageCollections.Count(); ++i)
731 count += iImageCollections[i]->RefCount();
740 void XSgDriverImpl::AllocMarkStart()
743 iHeap->__DbgMarkStart();
748 void XSgDriverImpl::AllocMarkEnd(TInt aCount)
751 TUint32 badCell = iHeap->__DbgMarkEnd(aCount);
755 _LIT(KPanicCategoryFormat, "SGALLOC:%08x");
757 category.Format(KPanicCategoryFormat, badCell);
758 User::Panic(category, 0);
763 void XSgDriverImpl::SetAllocFail(RAllocator::TAllocFail aType, TInt aRate)
766 iHeap->__DbgSetAllocFail(aType, aRate);
772 void XSgDriverImpl::AllocMarkStart()
777 void XSgDriverImpl::AllocMarkEnd(TInt /*aCount*/)
782 void XSgDriverImpl::SetAllocFail(RAllocator::TAllocFail /*aType*/, TInt /*aRate*/)
789 // MSgDriverAdapter::New()
791 EXPORT_C TInt MSgDriverAdapter::New(MSgDriverAdapter*& aPtr)
793 RHeap* heap = UserHeap::ChunkHeap(NULL, 0, KSgMaxLocalChunkSize);
798 XSgDriverImpl* driverImpl = static_cast<XSgDriverImpl*>(heap->Alloc(sizeof(XSgDriverImpl)));
804 new(driverImpl) XSgDriverImpl(heap);
805 TInt err = driverImpl->Construct();
808 driverImpl->Delete();