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.
16 #include "directgdipaniccodes.h" // Included first to resolve dependency
18 #include <graphics/sgimage.h>
19 #include <graphics/directgdiimagetarget.h>
20 #include <graphics/directgdidrawablesource.h>
21 #include <graphics/directgdipanics.h>
22 #include <graphics/directgdidriverinternal.h>
26 #include "directgdidriver.h"
27 #include "directgditypes.h"
28 #include "directgdidriverinternallibrary.inl"
30 typedef TInt (*TDriverCreateFunction)(CDirectGdiDriverInternal*&, RLibrary);
33 Returns the singleton instance of CDirectGdiDriver for the calling thread.
35 @return Pointer to an opened CDirectGdiDriver object else NULL if a CDirectGdiDriver had not been opened yet.
37 EXPORT_C CDirectGdiDriver* CDirectGdiDriver::Static()
39 return static_cast<CDirectGdiDriver*>(Dll::Tls());
43 Creates a new CDirectGdiDriver object if one has not already been created in this thread.
44 Performs any adaptation initialisation that is needed to allow the calling thread to use
45 functionality provided by the new DirectGDI interfaces. Calling this method multiple times
46 from the same client thread has no effect after the first call, other than to increase the
49 @see CDirectGdiDriver::Close()
52 @post The object reference counter is incremented. Adaptation resources for the
53 calling thread are created and initialised. The new DirectGDI interfaces are ready
54 to be used from the calling thread.
56 @return KErrNone if successful, otherwise one of the system-wide error codes.
58 EXPORT_C TInt CDirectGdiDriver::Open()
61 CDirectGdiDriver* self = Static();
63 // Check we're not already opened
66 self = new CDirectGdiDriver();
74 // Dynamically load DirectGDI adapter library.
75 // Under WINS two dlls will be available - directgdiadapter_sw.dll and directgdiadapter_vg.dll
76 // Need to select which dll to load. The value of the keyword SYMBIAN_GRAPHICS_DIRECTGDI_USE
77 // in the epoc.ini file will determine the dll to load.
78 // If this keyword does not appear in the epoc.ini file, the software version of
79 // directgdiadapter will be loaded by default.
80 _LIT(KVgAdapterDllName, "directgdiadapter_vg.dll");
81 _LIT(KSwAdapterDllName, "directgdiadapter_sw.dll");
82 TBool useOpenVg = EFalse;
83 UserSvr::HalFunction(EHalGroupEmulator, EEmulatorHalBoolProperty, (TAny*)"symbian_graphics_directgdi_use_openvg", &useOpenVg);
87 libraryName.Set(KVgAdapterDllName);
91 libraryName.Set(KSwAdapterDllName);
94 err = lib.Load(libraryName);
98 return (err==KErrNotFound ? KErrNotSupported : err);
101 // Create internal driver
102 err = self->CreateInternalDriver(lib);
110 // Under armv5, the appropriate dll (software or vg) is determined at rom build time.
111 err = self->CreateInternalDriver();
119 // Store a pointer to the new driver instance in thread local storage:
120 err = Dll::SetTls(self);
129 ++(self->iOpenCount);
136 Create the adaptation/internal object for this driver.
138 @pre The adaptation/internal driver object has not been created yet.
139 @post The adaptation/internal object has been created.
141 @return KErrNone if successful, otherwise one of the system-wide error codes.
144 TInt CDirectGdiDriver::CreateInternalDriver(const RLibrary& aLibrary)
146 // Function at ordinal 1 creates new CDirectGdiDriverInternal
147 TDriverCreateFunction create = reinterpret_cast<TDriverCreateFunction>(aLibrary.Lookup(1));
148 return create(iDriverInternal, aLibrary);
151 TInt CDirectGdiDriver::CreateInternalDriver()
153 return CDirectGdiDriverInternal::New(iDriverInternal, RLibrary());
158 Performs adaptation termination which is needed to release any internal resources
159 allocated by the calling thread. It will also perform an implicit Finish() by submitting
160 all outstanding requests from the calling thread. A DirectGDI client's thread that
161 uses the new DirectGDI interfaces must call this method at the end of its lifetime.
163 @see CDirectGdiDriver::Open()
165 @pre The driver is open from a call to Open().
166 @post The object open count is decremented. If the open count reaches zero, any
167 adaptation resources allocated to the calling thread must be released, and the CDirectGdiDriver
168 object for the calling thread is destroyed.
170 @panic DGDI 14, if Close() is called on a CDirectGdiDriver object that is not open.
172 EXPORT_C void CDirectGdiDriver::Close()
174 GRAPHICS_ASSERT_ALWAYS(iOpenCount > 0, EDirectGdiPanicDriverOpenCountError);
176 if (--iOpenCount == 0)
178 // We call Finish() instead of Flush() because it will decrease the reference count
179 // of images used in DrawResource() (if any) and is guaranteed that it will complete.
180 // Flush() cannot guarantee completion so we can't decrement image reference counts.
183 // Get the handle to the DLL used by iDriverInternal so that we can close
184 // it once iDriverInternal has been deleted.
185 RLibrary lib = iDriverInternal->Library();
189 // Free the space we saved for the driver singleton pointer in thead local storage
195 Ensures all outstanding requests on the calling thread are submitted for processing.
196 The method immediately returns and does not wait until all outstanding requests are
197 completely processed. Clients can continue issuing rendering requests after calling
200 @see CDirectGdiDriver::Finish()
202 @pre CDirectGdiDriver object has been initialised for the calling thread.
203 @post All outstanding requests have been submitted for processing.
205 EXPORT_C void CDirectGdiDriver::Flush()
207 iDriverInternal->Flush();
211 Ensures all outstanding rendering requests from the calling thread are submitted
212 and processed. This method will wait until all the outstanding rendering requests
213 have been completely processed.
215 @see CDirectGdiDriver::Flush()
217 @pre CDirectGdiDriver object has been initialised for the calling thread.
218 @post All outstanding requests have been completely processed.
220 EXPORT_C void CDirectGdiDriver::Finish()
222 iDriverInternal->Finish();
226 Returns the first error code (as the result of calling any DirectGDI API), if any,
227 since the last call to this function or, if it has not previously been called, since
228 the CDirectGdiDriver was initialised. Calling this function clears the error code.
230 @see CDirectGdiDriver::SetError(TInt)
232 @pre CDirectGdiDriver object has been initialised for the calling thread.
233 @post The error code has been reset after being read.
235 @return The first error code, if any, since the last call to this function or,
236 if it has not previously been called, since the CDirectGdiDriver was initialised.
237 KErrNone will indicate that no such error has occurred.
239 EXPORT_C TInt CDirectGdiDriver::GetError()
241 return iDriverInternal->GetError();
246 Sets the error code on the driver. If the error code is already set to a value other
247 than KErrNone, the error code will not be modified.
249 @see CDirectGdiDriver::GetError()
251 @param aErr The error code to set.
253 @pre CDirectGdiDriver object has been initialised for the calling thread.
254 @post The error code has been set.
256 EXPORT_C void CDirectGdiDriver::SetError(TInt aErr)
258 iDriverInternal->SetError(aErr);
262 Retrieves a pointer to an instance of the requested extension interface implementation, if provided.
264 @param aInterfaceId The globally unique identifier of the requested interface.
265 @param aInterface On return, holds the specified interface, or NULL if the interface cannot be found.
267 @pre CDirectGdiDriver object has been initialised for the calling thread.
269 @return KErrNone If the interface is supported, KErrExtensionNotSupported otherwise.
271 EXPORT_C TInt CDirectGdiDriver::GetInterface(TUid aInterfaceId, TAny*& aInterface)
273 return iDriverInternal->GetInterface(aInterfaceId, aInterface);
277 CDirectGdiDriver private constructor as this class is singleton.
280 @post Initialises the member reference counter to zero.
282 CDirectGdiDriver::CDirectGdiDriver()
287 CDirectGdiDriver default destructor.
292 @panic DGDI 15, if the resource count is not zero.
294 CDirectGdiDriver::~CDirectGdiDriver()
296 GRAPHICS_ASSERT_ALWAYS(iOpenCount == 0, EDirectGdiPanicDriverDestructorOpenCountError);
297 delete iDriverInternal;
301 Delegates the call to the CDirectGdiDriverInternal object, which creates a DirectGDI
302 adaptation-specific resource from the given drawable resource so it can be drawn
303 using the DirectGDI rendering API.
305 @see CloseDrawableSource()
307 @param aRDirectGdiDrawableSource The RDirectGdiDrawableSource object to be created
308 @param aRSgDrawable The RSgDrawable object to use when creating aRDirectGdiDrawableSource
310 @pre CDirectGdiDriver object has been initialised from the calling thread.
311 @post The DirectGDI adaptation-specific resource that is bound to the given
312 drawable resource is created and this handle is now associated with it. The reference
313 counter on the drawable resource is incremented. The CDirectGdiDriver for this thread
314 is now aware of and owns the adaptation specific resource.
316 @return KErrNone if successful, KErrNotSupported if the drawable resource is not
317 created with the correct usage (ESgUsageDirectGdiSource must be set), otherwise one of the system-wide error codes.
319 @panic DGDI 19, if this handle is already associated with a DirectGDI adaptation-specific drawable resource.
320 @panic DGDI 20, if the drawable resource is not valid.
322 TInt CDirectGdiDriver::CreateDrawableSource(RDirectGdiDrawableSource& aRDirectGdiDrawableSource, const RSgDrawable& aRSgDrawable)
324 GRAPHICS_ASSERT_ALWAYS(aRDirectGdiDrawableSource.Handle() == KNullHandle, EDirectGdiPanicDrawableSourceAlreadyExists);
326 if(aRSgDrawable.DrawableType() == KSgImageTypeUid)
327 {// make sure that drawable was created with the flag ESgUsageDirectGdiSource
328 RSgImage *image = (RSgImage*) &aRSgDrawable;
330 TInt err = image ->GetInfo (info);
331 GRAPHICS_ASSERT_ALWAYS(err == KErrNone, EDirectGdiPanicImageSourceInfoError);
333 if (!(info.iUsage & ESgUsageDirectGdiSource))
335 return KErrNotSupported;
338 return iDriverInternal->CreateDrawableSource(aRDirectGdiDrawableSource.iHandle, aRSgDrawable);
342 Delegates call to the CDirectGdiDriverInternal object, which destroys the DirectGDI
343 adaptation-specific resource associated with this handle. Calling this method on a
344 handle that is not associated with any DirectGDI adaptation specific resource will
345 do nothing. Once Close() is called, this handle can be reused.
347 @see CreateDrawableSource()
349 @param aSource The RDirectGdiDrawableSource object.
351 @pre CDirectGdiDriver object has been initialised from the calling thread.
352 @post The DirectGDI specific resource associated with this handle will be
353 destroyed (at any time preferred by the adaptation). This handle is no longer
354 associated with a DirectGDI specific resource. The reference counter of the
355 underlying non-image resource is decremented.
357 void CDirectGdiDriver::CloseDrawableSource(RDirectGdiDrawableSource& aSource)
359 iDriverInternal->CloseDrawableSource(aSource.iHandle);
363 Delegates call to the CDirectGdiDriverInternal object, which creates a DirectGDI
364 adaptation-specific resource from the given image resource so it can be used as a
365 target of DirectGDI rendering.
367 @see CloseImageTarget()
369 @param aTarget The RDirectGdiImageTarget object.
370 @param aImage The RSgImage object.
372 @pre CDirectGdiDriver object has been initialised from the calling thread.
373 The image resource has been fully constructed and created with the correct usage
374 that allows it to be used as a DirectGDI target.
376 @post The DirectGDI adaptation-specific resource that is bound to the given image
377 resource is created and this handle is now associated with it. The reference counter
378 on the image resource is incremented.
380 @return KErrNone if successful, KErrNotSupported if the image resource is not
381 created with the correct usage, otherwise one of the system-wide error codes.
383 @panic DGDI 17, if this handle is already associated with a DirectGDI adaptation-specific resource.
384 @panic DGDI 21, if the image resource is not valid.
386 TInt CDirectGdiDriver::CreateImageTarget(RDirectGdiImageTarget& aTarget, const RSgImage& aImage)
388 GRAPHICS_ASSERT_ALWAYS(aTarget.Handle() == KNullHandle, EDirectGdiPanicImageTargetAlreadyExists);
390 TInt err = aImage.GetInfo(info);
391 GRAPHICS_ASSERT_ALWAYS(err == KErrNone, EDirectGdiPanicImageTargetInfoError);
393 if (!(info.iUsage & ESgUsageDirectGdiTarget))
395 return KErrNotSupported;
397 return iDriverInternal->CreateImageTarget(aTarget.iHandle, aImage);
401 Delegates call to the CDirectGdiDriverInternal object, which destroys the DirectGDI
402 adaptation specific resource associated with this handle. Calling this method on a
403 handle that is not associated with any DirectGDI adaptation specific resource will
404 do nothing. Once Close() is called, this handle can be reused.
406 @see CreateImageTarget()
408 @param aTarget The RDirectGdiImageTarget object.
410 @pre CDirectGdiDriver object has been initialised from the calling thread.
412 @post The DirectGDI specific resource associated with this handle will be destroyed
413 (at any time preferred by the adaptation). This handle is no longer associated with
414 a DirectGDI specific resource. The reference counter of the underlying image
415 resource is decremented.
417 void CDirectGdiDriver::CloseImageTarget(RDirectGdiImageTarget& aTarget)
419 iDriverInternal->CloseImageTarget(aTarget.iHandle);
423 Delegates call to the CDirectGdiDriverInternal object, which creates a
424 MDirectGdiEngine object.
426 @param aEngine The MDirectGdiEngine object.
428 @pre CDirectGdiDriver object has been initialised from the calling thread.
431 @return KErrNone if successful, otherwise one of the system-wide error codes.
433 TInt CDirectGdiDriver::CreateEngine(MDirectGdiEngine*& aEngine)
435 return iDriverInternal->CreateEngine(aEngine);
439 Delegates call to the CDirectGdiDriverInternal object, which destroys a
440 MDirectGdiEngine object.
442 @param aEngine The MDirectGdiEngine object.
444 @pre CDirectGdiDriver object has been initialised from the calling thread.
447 void CDirectGdiDriver::DestroyEngine(MDirectGdiEngine* aEngine)
449 iDriverInternal->DestroyEngine(aEngine);