graphicsdeviceinterface/directgdiadaptation/swsrc/swdirectgdidriverimpl.cpp
author Faisal Memon <faisal.memon@nokia.com>
Thu, 06 May 2010 11:31:11 +0100
branchNewGraphicsArchitecture
changeset 47 48b924ae7197
parent 0 5d03bc08d59c
permissions -rw-r--r--
Applied patch 1, to provide a syborg specific minigui oby file. Need to compare this with the "stripped" version currently in the tree. This supplied version applies for Nokia builds, but need to repeat the test for SF builds to see if pruning is needed, or if the file needs to be device-specific.

// Copyright (c) 2007-2009 Nokia Corporation and/or its subsidiary(-ies).
// All rights reserved.
// This component and the accompanying materials are made available
// under the terms of "Eclipse Public License v1.0"
// which accompanies this distribution, and is available
// at the URL "http://www.eclipse.org/legal/epl-v10.html".
//
// Initial Contributors:
// Nokia Corporation - initial contribution.
//
// Contributors:
//
// Description:
//

#include "directgdiadapter.h"
#include "swdirectgdidriverimpl.h"
#include "swdirectgdiimagesourceimpl.h"
#include "swdirectgdiimagetargetimpl.h"
#include "pixelutil.h"
#include <e32cmn.h>
#include <pixelformats.h>
#include <graphics/sgimage_sw.h>

/**
@panic DGDIAdapter 6, if the reference count to DirectGDI driver is invalid.
@panic DGDIAdapter 45, if objects are still in the source or target image array.
*/
CSwDirectGdiDriverImpl::~CSwDirectGdiDriverImpl()
	{
	GRAPHICS_ASSERT_DEBUG(iEngines.Count() == 0, EDirectGdiPanicDriverInvalidRefCount);
	GRAPHICS_ASSERT_DEBUG(iTargets.Count() == 0, EDirectGdiPanicItemsLeftInImageArray);
	GRAPHICS_ASSERT_DEBUG(iImageSources.Count() == 0, EDirectGdiPanicItemsLeftInImageArray);
	
	// Delete all the engines owned by this driver
	iEngines.ResetAndDestroy();
	iTargets.ResetAndDestroy();
	iImageSources.ResetAndDestroy();
	}

/**
Constructor, which initialises base class.
*/
CSwDirectGdiDriverImpl::CSwDirectGdiDriverImpl(RLibrary aLibrary)
	: CDirectGdiDriverInternal(aLibrary)
	{
	}

/**
Empty function as this adaptation is synchronous.
@see CDirectGdiDriverInternal::Flush()
*/
void CSwDirectGdiDriverImpl::Flush()
	{
	}

/**
Empty function as this adaptation is synchronous.
@see CDirectGdiDriverInternal::Finish()
*/
void CSwDirectGdiDriverImpl::Finish()
	{
	}

/**
@see CDirectGdiDriver::GetError()
*/
TInt CSwDirectGdiDriverImpl::GetError()
	{
	TInt err = iErrorCode;
	iErrorCode = KErrNone;
	return err;
	}

/**
@see CDirectGdiDriverInternal::CloseDrawableSource()
@panic DGDIAdapter 57, if aHandleRet is not KNullHandle (debug only).
*/
TInt CSwDirectGdiDriverImpl::CreateDrawableSource(TInt& aHandleRet, const RSgDrawable& aSgDrawable)
	{
	TInt err = KErrNotSupported;
	const TUid typeUid = aSgDrawable.DrawableType(); 
	if(KSgImageTypeUid == typeUid)
		{
		CSwDirectGdiImageSourceImpl* imageSource = NULL;
		RSgImage* image =  (RSgImage*) (&aSgDrawable);
		err = CSwDirectGdiImageSourceImpl::New(imageSource, *this, *image);
	
		if (err == KErrNone)
			{
			aHandleRet = reinterpret_cast<TInt>(imageSource);
			}
		}
	return err;
	}
	
/**
@see CDirectGdiDriverInternal::CreateDrawableSource()
@panic DGDIAdapter 47, if the drawable source belonging to aHandle could not be found in the internal 
	   list of sources (debug only).
*/
void CSwDirectGdiDriverImpl::CloseDrawableSource(TInt& aHandle)
	{
	// Graphics Resource and DirectGDI reference implementation only support pixel based resource
	CSwDirectGdiImageSourceImpl* source = reinterpret_cast<CSwDirectGdiImageSourceImpl*>(aHandle);
	if (source)
		source->Close();
	aHandle = KNullHandle;
	}
	
/**
@see CDirectGdiDriverInternal::CreateImageTarget()
@panic DGDIAdapter 58, if aHandleRet is not KNullHandle (debug only).
@panic DGDIAdapter 1023, if the attempt to get an RSgImage interface MSgImage_Sw failed.
*/
TInt CSwDirectGdiDriverImpl::CreateImageTarget(TInt& aHandleRet, const RSgImage& aImage)
	{
	TSgImageInfo info;
	TInt ret = aImage.GetInfo(info);
	if (ret != KErrNone)
		{
		return ret;
		}
	TDisplayMode displayMode = PixelFormatUtil::ConvertToDisplayMode(info.iPixelFormat);
	TSize size = info.iSizeInPixels;
	CFbsDrawDevice* device = NULL;
	const MSgImage_Sw* pImage;
	TInt err = aImage.GetInterface(pImage);
	GRAPHICS_ASSERT_DEBUG((err == KErrNone) && (pImage != NULL), EDirectGdiPanicInvalidInterfaceHandle);
	TRAP(err, device = CFbsDrawDevice::NewBitmapDeviceL(size, displayMode, pImage->DataStride()));
	if (err != KErrNone)
		{		
		return err;
		}	
		
	// Setup the new device with the memory from the image
	device->SetBits(pImage->DataAddress());
		
	// Create a new image
	if (err == KErrNone)
		{
		CSwDirectGdiImageTargetImpl* target = NULL;
		err = CSwDirectGdiImageTargetImpl::New(target, *this, device, aImage);
		if (err == KErrNone)
			{
			// Make sure we aren't overwriting an exisiting valid handle.
			GRAPHICS_ASSERT_DEBUG(aHandleRet == KNullHandle, EDirectGdiPanicTargetHandleNotNull);
			aHandleRet = reinterpret_cast<TInt>(target);
			}		
		}
	
	if (err != KErrNone)
		{
		delete device;
		}
		
	return err;
	}
	
/**
@see CDirectGdiDriverInternal::CloseImageTarget()
@panic DGDIAdapter 40, if the image target associated with aHandle could not be found in the 
       internal list of targets (debug only).	
*/
void CSwDirectGdiDriverImpl::CloseImageTarget(TInt& aHandle)
	{
	CSwDirectGdiImageTargetImpl* target = reinterpret_cast<CSwDirectGdiImageTargetImpl*>(aHandle);
	if (target)
		target->Close();
	aHandle = KNullHandle;
	}
	
		
/**
@see CDirectGdiDriverInteral::CreateEngine()
*/
TInt CSwDirectGdiDriverImpl::CreateEngine(MDirectGdiEngine*& aEngine)
	{
	TInt err = KErrNoMemory; 
	CSwDirectGdiEngine* engine = new CSwDirectGdiEngine(this);
	aEngine = engine;	
	if(engine != NULL)	
		{		
		// Put this in an array so we can destroy it on close
		err = iEngines.InsertInAddressOrder(engine);
		if (KErrNone != err)
			{
			delete engine;
			aEngine = NULL;
			}
		}
	return err;
	}
		
/**
@see CDirectGdiDriverInternal::DestroyEngine()
@panic DGDIAdapter 60, if aEngine could not be found in the internal list of engines (debug only).
*/
void CSwDirectGdiDriverImpl::DestroyEngine(MDirectGdiEngine* aEngine)
	{
	TInt index = iEngines.FindInAddressOrder(aEngine);
	GRAPHICS_ASSERT_DEBUG(index != KErrNotFound, EDirectGdiPanicEngineNotFound);
	iEngines.Remove(index);
	delete (static_cast<CSwDirectGdiEngine*>(aEngine));
	}

/**
@see CDirectGdiDriverInternal::SetError()
*/
void CSwDirectGdiDriverImpl::SetError(TInt aErr)
	{
	if (aErr != KErrNone && iErrorCode == KErrNone)
		{
		iErrorCode = aErr;
		}
	}

/**
No extensions interfaces are supported.

@see CDirectGdiDriver::GetInterface()
 */
TInt CSwDirectGdiDriverImpl::GetInterface(TUid /*aInterfaceId*/, TAny*& aInterface)
	{
	aInterface = NULL;	
	return KErrExtensionNotSupported;			
	}

/**
Deactivate the target if it was the current target, decrement the reference count of the target as it can be shared 
across many DirectGDI contexts.

@param 	aRenderingTarget The target object which you want to deactivate.

@panic DGDIAdapter 55, if aRengeringTarget is NULL.
*/
void CSwDirectGdiDriverImpl::Deactivate(CSwDirectGdiImageTargetImpl* aRenderingTarget)
	{
	GRAPHICS_ASSERT_ALWAYS(aRenderingTarget, EDirectGdiPanicNullTargetDeactivate);
	aRenderingTarget->Close();
	}

/**
Checks whether the image source handle is valid. 

@param aHandle The handle to an Image Source that should be found.
@return ETrue if aHandle is a valid handle to an image registered with the driver, otherwise EFalse.
*/
TBool CSwDirectGdiDriverImpl::ValidImageSource(TInt aHandle) const
	{
	CSwDirectGdiImageSourceImpl* impl = reinterpret_cast<CSwDirectGdiImageSourceImpl*>(aHandle);
	return iImageSources.FindInAddressOrder(impl) != KErrNotFound;
	}

/**
Checks whether the target image handle is valid. 

@param aHandle The handle to an Image Target that should be found.
@return ETrue if aHandle is a valid handle to an image registered with the driver, otherwise EFalse.
*/
TBool CSwDirectGdiDriverImpl::ValidImageTarget(TInt aHandle) const
	{	
	CSwDirectGdiImageTargetImpl* impl = reinterpret_cast<CSwDirectGdiImageTargetImpl*>(aHandle);
	return iTargets.FindInAddressOrder(impl) != KErrNotFound;
	}

/**
Removes the specified image target from the internal list of targets.

@param aImageTarget The image to remove.
@return KErrNotFound if it could not be found in the internal list, otherwise KErrNone.
*/
TInt CSwDirectGdiDriverImpl::UnregisterTargetImage(const CSwDirectGdiImageTargetImpl& aImageTarget)
	{
	TInt err = KErrNone;
	TInt index = iTargets.FindInAddressOrder(&aImageTarget);
	if (index == KErrNotFound)
		{
		err = KErrNotFound;
		}
	else
		{
		iTargets.Remove(index);
		iTargets.GranularCompress();
		}
	return err;
	}

/**
Removes the specified image source from the internal list of sources.

@param aImageSource The image to remove.
@return KErrNotFound if it could not be found in the internal list, otherwise KErrNone.
*/
TInt CSwDirectGdiDriverImpl::UnregisterSourceImage(const CSwDirectGdiImageSourceImpl& aImageSource)
	{
	TInt err = KErrNone;
	TInt index = iImageSources.FindInAddressOrder(&aImageSource);
	if (index == KErrNotFound)
		{		
		err = KErrNotFound;
		}
	else
		{	
		iImageSources.Remove(index);
		iImageSources.GranularCompress();
		}
	return err;
	}

/**
Helper function. When an image source is constructed, it registers itself on the driver's internal 
list of image sources.
@param aImageSource The image to add to the array.
@return KErrNone if successful, otherwise one of the system-wide error codes.
*/
TInt CSwDirectGdiDriverImpl::RegisterSourceImage(const CSwDirectGdiImageSourceImpl& aImageSource)
	{
	return iImageSources.InsertInAddressOrder(&aImageSource);
	}

/**
Helper function. When an image target is constructed, it registers itself on the driver's internal 
list of image targets.
@param aImageTarget The image to add to the array.
@return KErrNone if successful, otherwise one of the system-wide error codes.
*/
TInt CSwDirectGdiDriverImpl::RegisterTargetImage(const CSwDirectGdiImageTargetImpl& aImageTarget)
	{
	return iTargets.InsertInAddressOrder(&aImageTarget);
	}