diff -r 000000000000 -r 2e3d3ce01487 appfw/apparchitecture/apfile/apfmimecontentpolicy.cpp --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/appfw/apparchitecture/apfile/apfmimecontentpolicy.cpp Tue Feb 02 10:12:00 2010 +0200 @@ -0,0 +1,526 @@ +// Copyright (c) 2002-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 FILES +#include +#include // RFs +#include +#include +#include +#include +#include +#include // For RApaLsSession + +// Resource file name. +_LIT(KCEResourceFile, "z:\\resource\\apps\\apfmimecontentpolicy.rsc"); + +// This is needed for resource reading. +const TInt KCCMask(0x00000fff); + + +NONSHARABLE_CLASS(CApfMimeContentPolicyImpl) : public CBase + { +public: // Constructors and destructor + static CApfMimeContentPolicyImpl* NewL(); + static CApfMimeContentPolicyImpl* NewL(RFs& aFs); + ~CApfMimeContentPolicyImpl(); + +public: // New functions + TBool IsClosedType(const TDesC& aMimeType) const; + TBool IsClosedExtension(const TDesC& aFileExtension) const; + TBool IsClosedFileL(const TDesC& aFileName) const; + TBool IsDRMEnvelopeL(const TDesC& aFileName) const; + TBool IsClosedFileL(RFile& aFileHandle) const; + TBool IsDRMEnvelopeL(RFile& aFileHandle) const; + +private: + CApfMimeContentPolicyImpl(); + void ConstructL(); + void ConstructL(RFs& aFs); + TBool IsClosedFileL(RFile& aFileHandle, const TDesC& aFileName) const; + void ReadResourcesL(RFs& aFs); + +private: + CDesCArrayFlat* iCcl; // Closed content list. + CDesCArrayFlat* iExtList; // Closed extensions list. + RApaLsSession iLs; // A session to the Application Architecture server. + mutable RFs iFs; // File session + TBool iFsConnected; // ETrue if connected to file server, else EFalse + }; + + +// +// class CApfMimeContentPolicy +// + +/** +C++ default constructor. +*/ +CApfMimeContentPolicy::CApfMimeContentPolicy() + { + // Nothing to do here. + } + +/** +By default Symbian 2nd phase constructor is private. +*/ +void CApfMimeContentPolicy::ConstructL() + { + iImpl = CApfMimeContentPolicyImpl::NewL(); + } + +/** +By default Symbian 2nd phase constructor is private. +@param aFs A handle to a shared file server session. +*/ +void CApfMimeContentPolicy::ConstructL(RFs& aFs) + { + iImpl = CApfMimeContentPolicyImpl::NewL(aFs); + } + +/** +Creates a new Mime Content Policy object, and puts a pointer to it onto the cleanup stack. +@return The new Mime Content Policy object. +*/ +EXPORT_C CApfMimeContentPolicy* CApfMimeContentPolicy::NewLC() + { + CApfMimeContentPolicy* self = new (ELeave) CApfMimeContentPolicy(); + CleanupStack::PushL(self); + self->ConstructL(); + return self; + } + +/** +Creates a new Mime Content Policy object, and puts a pointer to it onto the cleanup stack. +@param aFs A handle to a shared file server session. +@return The new Mime Content Policy object. +*/ +EXPORT_C CApfMimeContentPolicy* CApfMimeContentPolicy::NewLC(RFs& aFs) + { + CApfMimeContentPolicy* self = new (ELeave) CApfMimeContentPolicy(); + CleanupStack::PushL(self); + self->ConstructL(aFs); + return self; + } + +/** +Creates a new Mime Content Policy object. +@return The new Mime Content Policy object. +*/ +EXPORT_C CApfMimeContentPolicy* CApfMimeContentPolicy::NewL() + { + CApfMimeContentPolicy* self = NewLC(); + CleanupStack::Pop(); + return self; + } + +/** +Creates a new Mime Content Policy object. +@param aFs A handle to a shared file server session. +@return The new Mime Content Policy object. +*/ +EXPORT_C CApfMimeContentPolicy* CApfMimeContentPolicy::NewL(RFs& aFs) + { + CApfMimeContentPolicy* self = NewLC(aFs); + CleanupStack::Pop(); + return self; + } + + +/** +Destructor. +*/ +EXPORT_C CApfMimeContentPolicy::~CApfMimeContentPolicy() + { + delete iImpl; + } + +/** +Checks if given MIME type is included in closed content list. +@param aMimeType The mime type to be checked. +@return ETrue if given MIME type is in closed content list else returns EFalse. +*/ +EXPORT_C TBool CApfMimeContentPolicy::IsClosedType(const TDesC& aMimeType) + { + return iImpl->IsClosedType(aMimeType); + } + +/** +Checks the extension of given file against list of closed file extensions. +@param aFileExtension File extension to be checked. +@return ETrue if extension of given file name is in closed extensions list else returns EFalse. +*/ +EXPORT_C TBool CApfMimeContentPolicy::IsClosedExtension(const TDesC& aFileExtension) + { + return iImpl->IsClosedExtension(aFileExtension); + } + +/** +Checks if given file is Closed or not. +This method checks for forward lock and superdistribution statuses of the file, in addition to IsClosedExtension +and IsClosedType checks. + +@param aFileName A file to be checked +@return ETrue if given file is closed else returns EFalse. +@leave KErrNone, if successful; otherwise one of the other system-wide error codes +*/ +EXPORT_C TBool CApfMimeContentPolicy::IsClosedFileL(const TDesC& aFileName) + { + return iImpl->IsClosedFileL(aFileName); + } + + +/** +Checks if given file is a DRM envelope. Can leave if file handling fails. +@param aFileName A file to be checked. +@return ETrue if file is DRM envelope else returns EFalse. +@leave KErrCANotSupported if the requested attribute does not exist. +KErrPermissionDenied if the access to the protected content is not permitted +by the CAF Agent. Otherwise one of the other CAF error codes defined in +caferr.h or one of the other system-wide error codes for any other errors. +*/ +EXPORT_C TBool CApfMimeContentPolicy::IsDRMEnvelopeL(const TDesC& aFileName) + { + return iImpl->IsDRMEnvelopeL(aFileName); + } + +/** +Checks if given file is Closed or not. This method checks for forward lock and +superdistribution statuses of the file, in addition to IsClosedExtension and IsClosedType checks. +Remember to make a file handle sharable. When a file handle is shared, the RFs handle has to be shared too. +@param aFileHandle Handle to the file to be checked. +@return ETrue if given file is closed else returns EFalse. +@leave KErrNone, if successful; KErrBadHandle if an invalid handle has been passed as a parameter. +otherwise one of the other system-wide error codes +*/ +EXPORT_C TBool CApfMimeContentPolicy::IsClosedFileL(RFile& aFileHandle) + { + return iImpl->IsClosedFileL(aFileHandle); + } + + +/** +Checks if given file is a DRM envelope. Can leave if file handling fails. +@param aFileHandle Handle to the file to be checked. +@return ETrue if file is DRM envelope else returns EFalse. +@leave KErrCANotSupported if the requested attribute does not exist. +KErrPermissionDenied if the access to the protected content is not permitted +by the CAF Agent. Otherwise one of the other CAF error codes defined in +caferr.h or one of the other system-wide error codes for any other errors. +*/ +EXPORT_C TBool CApfMimeContentPolicy::IsDRMEnvelopeL(RFile& aFileHandle) + { + return iImpl->IsDRMEnvelopeL(aFileHandle); + } + +// +// class CApfMimeContentPolicyImpl +// + +/** +C++ default constructor. +*/ +CApfMimeContentPolicyImpl::CApfMimeContentPolicyImpl() + { + // C++ default constructor can NOT contain any code, that + } + +/** +By default Symbian 2nd phase constructor is private. +*/ +void CApfMimeContentPolicyImpl::ConstructL() + { + // Resource reading is done without coe & eikon env. + User::LeaveIfError(iFs.Connect()); + iFsConnected = ETrue; + + User::LeaveIfError(iFs.ShareProtected()); + ReadResourcesL(iFs); + } + +/** +By default Symbian 2nd phase constructor is private. +@param aFs A handle to a shared file server session. +*/ +void CApfMimeContentPolicyImpl::ConstructL(RFs& aFs) + { + iFsConnected = EFalse; + iFs = aFs; + ReadResourcesL(iFs); + } + +/** +Two-phased constructor. +*/ +CApfMimeContentPolicyImpl* CApfMimeContentPolicyImpl::NewL() + { + CApfMimeContentPolicyImpl* self = new (ELeave) CApfMimeContentPolicyImpl(); + CleanupStack::PushL(self); + self->ConstructL(); + CleanupStack::Pop(self); + + return self; + } + +/** +Two-phased constructor. +*/ +CApfMimeContentPolicyImpl* CApfMimeContentPolicyImpl::NewL(RFs& aFs) + { + CApfMimeContentPolicyImpl* self = new (ELeave) CApfMimeContentPolicyImpl(); + CleanupStack::PushL(self); + self->ConstructL(aFs); + CleanupStack::Pop(self); + + return self; + } + +/** +Destructor. +*/ +CApfMimeContentPolicyImpl::~CApfMimeContentPolicyImpl() + { + iLs.Close(); + + if(iFsConnected) + iFs.Close(); + + delete iCcl; + delete iExtList; + } + +/** +Checks if given MIME type is included in closed content list. +@param aMimeType The mime type to be checked. +@return ETrue if given MIME type is in closed content list else returns EFalse. +*/ +TBool CApfMimeContentPolicyImpl::IsClosedType(const TDesC& aMimeType) const + { + TInt dummy = 0; + + // Check if given descriptor is in closed content list. + // Find() returns 0 if found from array, non-zero if not. + const TBool found = (iCcl->FindIsq(aMimeType, dummy) == 0); + return found; + } + +/** +Checks the extension of given file against list of closed file extensions. +@param aFileExtension File extension to be checked. +@return ETrue if extension of given file name is in closed extensions list else returns EFalse. +*/ +TBool CApfMimeContentPolicyImpl::IsClosedExtension(const TDesC& aFileExtension) const + { + TInt dummy = 0; + + // Check if given descriptor is in closed content list. + // Find() returns 0 if found from array, non-zero if not. + const TBool found = (iExtList->FindIsq(aFileExtension, dummy) == 0); + return found; + } + +/** +Checks if given file is Closed or not. This method checks for forward lock and superdistribution statuses +of the file, in addition to IsClosedExtension and IsClosedType checks. +@param aFileName A file to be checked. +@return ETrue if given file is closed else returns EFalse. +@leave KErrNone, if successful; otherwise one of the other system-wide error codes. +*/ +TBool CApfMimeContentPolicyImpl::IsClosedFileL(const TDesC& aFileName) const + { + RFile file; + TInt protectionError; + + // open file to be checked for protection + protectionError = file.Open(iFs, aFileName, EFileRead | EFileShareReadersOnly); + + // If the file is already opened using EFileShareAny (which is against best practices), we have to use it also. + if (protectionError == KErrInUse) + protectionError = file.Open(iFs, aFileName, EFileRead | EFileShareAny); + + User::LeaveIfError(protectionError); + + CleanupClosePushL(file); + const TBool retVal = IsClosedFileL(file, aFileName); + CleanupStack::PopAndDestroy(); // file + return retVal; + } + +/** +Checks if given file is Closed or not. This method checks for forward lock and superdistribution statuses +of the file, in addition to IsClosedExtension and IsClosedType checks. +@param aFileHandle Handle to the file to be checked. +@return ETrue if given file is closed else returns EFalse. +@leave KErrNone, if successful; KErrBadHandle if an invalid handle has been passed as a parameter. +otherwise one of the other system-wide error codes. +*/ +TBool CApfMimeContentPolicyImpl::IsClosedFileL(RFile& aFileHandle) const + { + if (aFileHandle.SubSessionHandle()) + { + TFileName name; + aFileHandle.Name(name); + return IsClosedFileL(aFileHandle, name); + } + + User::Leave(KErrBadHandle); + return EFalse; // to keep compiler happy + } + +/** +Checks if given file is Closed or not. +@param aFileHandle Handle to the file to be checked. +@param aFileName File to be checked. +@return ETrue if given file is closed else returns EFalse. +@leave KErrNone, if successful; KErrBadHandle if an invalid handle has been passed as a parameter. +otherwise one of the other system-wide error codes. +*/ +TBool CApfMimeContentPolicyImpl::IsClosedFileL(RFile& aFileHandle, const TDesC& aFileName) const + { + TInt value = 0; + // allocate a content object + ContentAccess::CContent* content = ContentAccess::CContent::NewL(aFileHandle); + + // Check if file is protected + // If the file type can not be determined just forget it. + if (content->GetAttribute(ContentAccess::EIsProtected, value) == KErrNone) + { + if (value) + { + // File is DRM protected + value = 0; + // Check if file is forwardable + if (content->GetAttribute(ContentAccess::EIsForwardable, value) == KErrNone) + { + delete content; + if (!value) + { + // EIsProtected == ETrue && EIsForwardable == EFalse => EForwardLocked + // If forwardlocked, sending never allowed, so return without further ado. + return ETrue; + } + else + { + // EIsProtected == ETrue && EIsForwardable == ETrue => ESuperDistributable + // No need to check extension or mime type for files that are superdistributable, + // they must not be blocked in any case. + return EFalse; + } + } + } + } + + delete content; + content = NULL; + + // Check file extension. + TParse parser; + parser.Set(aFileName, NULL, NULL); + if (IsClosedExtension(parser.Ext())) + return ETrue; + + // Recognize and check MIME type. + TDataType recData; + TUid uid; + const TInt err(iLs.AppForDocument(aFileHandle, uid, recData)); + User::LeaveIfError(err); + + return IsClosedType(recData.Des()); // Check MIME type. + } + +/** +Checks if given file is a DRM envelope. Can leave if file handling fails. +@param aFileName A file to be checked. +@return ETrue if file is DRM envelope else returns EFalse. +@leave KErrCANotSupported if the requested attribute does not exist. +KErrPermissionDenied if the access to the protected content is not permitted +by the CAF Agent. Otherwise one of the other CAF error codes defined in +caferr.h or one of the other system-wide error codes for any other errors. +*/ +TBool CApfMimeContentPolicyImpl::IsDRMEnvelopeL(const TDesC& aFileName) const + { + // allocate a content object + ContentAccess::CContent* content = ContentAccess::CContent::NewLC(aFileName, ContentAccess::EContentShareReadOnly); + + TInt value = 0; + User::LeaveIfError(content->GetAttribute(ContentAccess::EIsProtected, value)); + CleanupStack::PopAndDestroy(content); + + return (value != 0); // File is DRM protected if value != 0 + } + +/** +Checks if given file is a DRM envelope. Can leave if file handling fails. +@param aFileHandle Handle to the file to be checked. +@return ETrue if file is DRM envelope else returns EFalse. +@leave KErrCANotSupported if the requested attribute does not exist. +KErrPermissionDenied if the access to the protected content is not permitted +by the CAF Agent. Otherwise one of the other CAF error codes defined in +caferr.h or one of the other system-wide error codes for any other errors. +*/ +TBool CApfMimeContentPolicyImpl::IsDRMEnvelopeL(RFile& aFileHandle) const + { + // allocate a content object + ContentAccess::CContent* content = ContentAccess::CContent::NewLC(aFileHandle); + + TInt value = 0; + User::LeaveIfError(content->GetAttribute(ContentAccess::EIsProtected, value)); + CleanupStack::PopAndDestroy(content); + + return (value != 0); // File is DRM protected if value != 0 + } + + +/** +Reads closed content list and closed extensions list. Connects to RApaLsSession. +Called by constructor. +@param aFs A handle to a shared file server session. +*/ +void CApfMimeContentPolicyImpl::ReadResourcesL(RFs& aFs) + { + TResourceReader reader; + + // Resource reading is done without coe & eikon env. + RResourceFile rsFile; + rsFile.OpenL(aFs, KCEResourceFile); + CleanupClosePushL(rsFile); + + // Read closed content list. + // Remove offset from id + HBufC8* rBuffer = rsFile.AllocReadLC(R_COMMONENG_CLOSED_CONTENT_LIST & KCCMask); + reader.SetBuffer(rBuffer); + ASSERT(!iCcl); + iCcl = reader.ReadDesCArrayL(); + CleanupStack::PopAndDestroy(rBuffer); // rBuffer + + // Read closed extensions list. + // Remove offset from id + rBuffer = rsFile.AllocReadLC(R_COMMONENG_CLOSED_EXTENSIONS_LIST & KCCMask); + reader.SetBuffer(rBuffer); + ASSERT(!iExtList); + iExtList = reader.ReadDesCArrayL(); + CleanupStack::PopAndDestroy(2); // rBuffer, rsFile + + // Sort lists to enable binary find + iCcl->Sort(); + iExtList->Sort(); + + // Connect RApaLsSession and leave it open for the whole + // lifetime of the object. + User::LeaveIfError(iLs.Connect()); + } +