crashanalysercmd/Libraries/Engine/CrashItemLib/Engine/ProblemDetectors/Registers/CIPDRegAvailability.cs
author Matti Laitinen <matti.t.laitinen@nokia.com>
Thu, 11 Feb 2010 15:50:58 +0200
changeset 0 818e61de6cd1
permissions -rw-r--r--
Add initial version of Crash Analyser cmdline under EPL

/*
* Copyright (c) 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:
*
*/
using System;
using System.Text;
using System.Collections.Generic;
using CrashItemLib.Crash.Container;
using CrashItemLib.Crash.Stacks;
using CrashItemLib.Crash.Registers;
using CrashItemLib.Crash.Base;
using CrashItemLib.Crash.Messages;
using CrashItemLib.Crash.Summarisable;
using SymbianStructuresLib.Arm.Registers;
using SymbianUtils.Range;

namespace CrashItemLib.Engine.ProblemDetectors.Registers
{
    internal class CIPDRegAvailability : CIProblemDetector
    {
        #region Constructors
        public CIPDRegAvailability()
		{
        }
		#endregion

        #region From CIProblemDetector
        public override void Check( CIContainer aContainer )
        {
            CISummarisableEntityList list = aContainer.Summaries;
            foreach ( CISummarisableEntity entry in list )
            {
                // Check that each stack contains some registers and at least the SP.
                bool stackAvailable = entry.IsAvailable( CISummarisableEntity.TElement.EElementStack );
                bool regsAvailable = entry.IsAvailable( CISummarisableEntity.TElement.EElementRegisters );
                //
                if ( stackAvailable )
                {
                    CIStack stack = entry.Stack;
                    //
                    if ( regsAvailable )
                    {
                        CIRegisterList regs = stack.Registers;

                        // Check that SP, LR and PC are available
                        bool pointerAvailable = regs.Contains( TArmRegisterType.EArmReg_SP );
                        if ( !pointerAvailable )
                        {
                            base.CreateWarning( aContainer, stack,
                               LibResources.CIPDRegAvailability_MissingSP_Title,
                               string.Format( LibResources.CIPDRegAvailability_MissingSP_Description, base.CreateIdentifierText( entry ) )
                               );
                        }
                        //
                        bool lrAvailable = regs.Contains( TArmRegisterType.EArmReg_LR );
                        if ( !lrAvailable )
                        {
                            base.CreateWarning( aContainer, stack,
                               LibResources.CIPDRegAvailability_MissingLR_Title,
                               string.Format( LibResources.CIPDRegAvailability_MissingLR_Description, base.CreateIdentifierText( entry ) )
                               );
                        }
                        //
                        bool pcAvailable = regs.Contains( TArmRegisterType.EArmReg_PC );
                        if ( !pcAvailable )
                        {
                            base.CreateWarning( aContainer, stack,
                               LibResources.CIPDRegAvailability_MissingPC_Title,
                               string.Format( LibResources.CIPDRegAvailability_MissingPC_Description, base.CreateIdentifierText( entry ) )
                               );
                        }
                     
                        // If R0 is available, check if it is 0 and check whether an exception occurred - if so, it was possibly
                        // caused by de-referencing a NULL this pointer.
                        bool threadAvailable = entry.IsAvailable( CISummarisableEntity.TElement.EElementThread );
                        if ( threadAvailable )
                        {
                            if ( regs.Contains( TArmRegisterType.EArmReg_00 ) )
                            {
                                CIRegister r0 = regs[ TArmRegisterType.EArmReg_00 ];
                                //
                                bool r0WasNull = ( r0.Value == 0 );
                                bool wasException = entry.IsAbnormalTermination && ( entry.Thread.ExitInfo.Type == CrashItemLib.Crash.ExitInfo.CIExitInfo.TExitType.EExitTypeException );
                                bool wasKernExec3 = entry.IsAbnormalTermination && ( entry.Thread.ExitInfo.Type == CrashItemLib.Crash.ExitInfo.CIExitInfo.TExitType.EExitTypePanic && 
                                                                                     entry.Thread.ExitInfo.Category.ToUpper() == "KERN-EXEC" && 
                                                                                     entry.Thread.ExitInfo.Reason == 3 );
                                //
                                if ( r0WasNull && ( wasException || wasKernExec3 ) )
                                {
                                    base.CreateWarning( aContainer, r0,
                                       LibResources.CIPDRegAvailability_NullThisPointer_Title,
                                       string.Format( LibResources.CIPDRegAvailability_NullThisPointer_Description, base.CreateIdentifierText( entry ) )
                                       );
                                }
                            }
                        }
                    }
                    else
                    {
                        base.CreateWarning( aContainer, stack, 
                               LibResources.CIPDRegAvailability_NoRegsForStack_Title,
                               string.Format( LibResources.CIPDRegAvailability_NoRegsForStack_Description, base.CreateIdentifierText( entry ) )
                               );
                    }
                }
            }
        }
        #endregion

        #region Properties
        #endregion

        #region Internal methods
        #endregion

        #region From System.Object
        #endregion

        #region Data members
		#endregion
    }
}