appfw/apparchitecture/apserv/apsnnapps.h
author Dremov Kirill (Nokia-D-MSW/Tampere) <kirill.dremov@nokia.com>
Fri, 12 Mar 2010 15:43:21 +0200
branchRCL_3
changeset 12 7645e9ce10dc
parent 0 2e3d3ce01487
permissions -rw-r--r--
Revision: 201007 Kit: 201008

// 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:
// apsnnapps.h
//

#ifndef APSNNAPPS_H
#define APSNNAPPS_H

#include <e32base.h>
#include <f32file.h>
#include <s32file.h>
#include <badesca.h>	// typedef CDesCArray

#include "apsnnappupdates.h"

// classes referenced
class CApaAppArcServer;
class CApsNonNativeApplicationsUpdateList;

_LIT(KLitPathForTemporaryNonNativeResourceAndIconFiles, "\\private\\10003a3f\\temp\\NonNativeUpdates\\");

/**
@internalComponent
*/
NONSHARABLE_CLASS(CApsNonNativeApplicationsManager) : public CBase
	{
public:
	static CApsNonNativeApplicationsManager* NewL(CApaAppArcServer& aServ, RFs& aFs);
	~CApsNonNativeApplicationsManager();

private:
	CApsNonNativeApplicationsManager(CApaAppArcServer& aServ, RFs& aFs);

public:
	void NotifyScanComplete();
	void PrepareNonNativeApplicationsUpdatesL();
	void RegisterNonNativeApplicationL(const RMessage2& aMessage);
	void DeregisterNonNativeApplicationL(const RMessage2& aMessage);
#ifdef _DEBUG
	void ForceFailInNonNativeApplicationsUpdatesL();
	void ForcePanicInNonNativeApplicationsUpdatesL();
	void ForcePanicInNonNativeApplicationsRollbackL();
#endif
	void CommitNonNativeApplicationsUpdatesL(const RMessage2& aMessage);
	void RollbackNonNativeApplicationsUpdates();
	static void RecoverFromUpdateLogL(RFs& aFs);

private:
	void DoRegisterNonNativeApplicationL(const RMessage2& aMessage);
	void AbortNonNativeApplicationsUpdates();
	static void StaticAbortNonNativeApplicationsUpdates(TAny* aSelf);
	void CheckForUpdateAppsLockL();

private:
	CApaAppArcServer& iServ;
	RFs& iFs;
	RMessage2 iNotifyOnScanCompleteMsg;
	TBool iNonNativeApplicationsUpdateAppsLock;
	CApsNonNativeApplicationsUpdateList* iUpdateList;
	};

class RApsUpdateLog;

/**
This class holds a list of updates to the non-native apps registry that should
be performed as one transaction, along with code to execute / rollback the updates.



Update list structure:

These are the possible valid types of list:
(* means "can occur 0 or more times", ? means "can occur 0 or 1 times")

EPerformed*, EPerforming?, ENew*
EPerformed*, ERollingBack?, ERolledBack*, ENew*

The list can be destructed in any of these states:

EPerformed*, ERolledBack*, ENew*

@internalComponent
*/
NONSHARABLE_CLASS(CApsNonNativeApplicationsUpdateList) : public CBase
	{
public:
	static CApsNonNativeApplicationsUpdateList* NewL(RFs& aFs);
	static void RecoverFromUpdateLogL(RFs& aFs, RFile& aFile);
	~CApsNonNativeApplicationsUpdateList();

private:
	CApsNonNativeApplicationsUpdateList(RFs& aFs);
	void RecoverFromUpdateLogL(RFile& aFile);

public:
	void Append(CApsNonNativeApplicationsUpdate* aUpdate);
	void ExecuteL();

private:
	void Rollback(CApsNonNativeApplicationsUpdate* aStartPoint, RApsUpdateLog& aUpdateLog);
	void SetLogWriteStream(RWriteStream* aStream);
	void InternalizeL(RReadStream& aStream, TInt& aPosition);
	void InternalizeActionL(RReadStream& aStream, int& aPosition);
	void InternalizeNewUpdateL(RReadStream& aStream, TInt& aPosition, CApsNonNativeApplicationsUpdate::TLogUpdateType aType, TUid aUid);
	void InternalizePerformUpdateL(RReadStream& aStream, TInt& aPosition, CApsNonNativeApplicationsUpdate::TLogUpdateType aType, TUid aUid);
	void InternalizeRollbackUpdateL(RReadStream& aStream, TInt& aPosition, CApsNonNativeApplicationsUpdate::TLogUpdateType aType, TUid aUid);

private:
	RFs& iFs;
	CApsNonNativeApplicationsUpdate* iUpdateListHead;
	CApsNonNativeApplicationsUpdate* iLogReplayCurrentUpdate;
	TBool iEEndOfUpdateRequiredToFixLog; // for recovering from the log. This is ETrue if we've reached a beta stopping point.
	};


/*
This class opens & owns the recovery log file

Recovery log structure:

As updates are performed, they are written to a log. The log's format on disk is documented here.

When we read the log in, there are certain points that are considered to be "valid stopping points"
These points are at the end of every useful section of data. When we keep track of the stream-position
during internalization, we make sure that the position we record is always at one of these points, never
between them.
This means that we can then truncate the log to the last of these points and then append new operations to the
log.
There are two types of stopping points, i've called them alpha and beta. At an alpha stopping point we
can start appending new operations to the log immediately. At a beta stopping point, we must append a
EEndOfUpdate marker first.


These points are denoted in the structure below with "--- stopping point ---"

LogStructure =
{
	(NewUpdateSection,
	PerformUpdateSection)*
	RollbackUpdateSection*
}

NewUpdateSection = 
{
	Int8  ENewUpdate
	Int8  TLogUpdateType
	Int32 TUid
	... dependent on update type ...
	--- alpha stopping point ---
}

PerformUpdateSection =
{
	Int8  EPerformUpdate
	Int8  TLogUpdateType
	Int32 TUid
	--- beta stopping point ---
	(UpdateContinuationSection | UpdateStateChangeSection)*
	Int8  EEndOfUpdate
	--- alpha stopping point ---
}

RollbackUpdateSection =
{
	Int8  ERollbackUpdate
	Int8  TLogUpdateType
	Int32 TUid
	--- beta stopping point ---
	(UpdateContinuationSection | UpdateStateChangeSection)*
	Int8  EEndOfUpdate
	--- alpha stopping point ---
}

UpdateContinuationSection =
{
	Int8 EUpdateContinuation
	... dependent on update type ...
	--- beta stopping point ---
}

UpdateStateChangeSection =
{
	Int8 EChangeOfUpdateState
	Int8 TState
	--- beta stopping point ---
}

@internalComponent
*/
NONSHARABLE_CLASS(RApsUpdateLog) 
	{
public:
	RApsUpdateLog(RFs& aFs);
	void OpenL();
	void Open(RFile& aFile, TUint aSeekPos);
	void Close();
	CDesCArray& NewRegistrationFiles();
	CDesCArray& DrivesAffected();
	RWriteStream& LogWriteStream();
private:
	RFs& iFs;
	CDesCArray* iFilesRegistered;
	CDesCArray* iDrivesAffected;
	RFileWriteStream iLogWriteStream;
	TFileName iLogFileName;
	};

#endif // APSNNAPPS_H