/* Copyright (C) 2011-2015 de4dot@gmail.com This file is part of de4dot. de4dot is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. de4dot is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with de4dot. If not, see . */ using System; using System.Collections.Generic; namespace de4dot.blocks { class DeadBlocksRemover { MethodBlocks methodBlocks; Dictionary checkedBaseBlocks = new Dictionary(); Dictionary checkedScopeBlocks = new Dictionary(); Stack baseBlocksToCheck = new Stack(); Stack scopeBlocksToCheck = new Stack(); public DeadBlocksRemover(MethodBlocks methodBlocks) { this.methodBlocks = methodBlocks; } public int Remove() { AddScopeBlock(methodBlocks); ProcessAll(); return RemoveDeadBlocks(); } class ScopeBlockInfo { public ScopeBlock scopeBlock; public IList deadBlocks = new List(); public ScopeBlockInfo(ScopeBlock scopeBlock) { this.scopeBlock = scopeBlock; } } int RemoveDeadBlocks() { int numDeadBlocks = 0; var infos = new Dictionary(); var deadBlocksDict = new Dictionary(); foreach (var baseBlock in FindDeadBlocks()) { deadBlocksDict[baseBlock] = true; ScopeBlock parent = baseBlock.Parent; ScopeBlockInfo info; if (!infos.TryGetValue(parent, out info)) infos[parent] = info = new ScopeBlockInfo(parent); info.deadBlocks.Add(baseBlock); numDeadBlocks++; } foreach (var info in infos.Values) info.scopeBlock.RemoveAllDeadBlocks(info.deadBlocks, deadBlocksDict); return numDeadBlocks; } IList FindDeadBlocks() { var deadBlocks = new List(); foreach (var bb in methodBlocks.GetAllBaseBlocks()) { if (!checkedBaseBlocks.ContainsKey(bb)) deadBlocks.Add(bb); } return deadBlocks; } void AddScopeBlock(ScopeBlock scopeBlock) { scopeBlocksToCheck.Push(scopeBlock); } void ProcessAll() { bool didSomething; do { didSomething = false; while (baseBlocksToCheck.Count > 0) { ProcessBaseBlock(baseBlocksToCheck.Pop()); didSomething = true; } while (scopeBlocksToCheck.Count > 0) { ProcessScopeBlock(scopeBlocksToCheck.Pop()); didSomething = true; } } while (didSomething); } void ProcessBaseBlock(BaseBlock baseBlock) { if (baseBlock == null || checkedBaseBlocks.ContainsKey(baseBlock)) return; checkedBaseBlocks[baseBlock] = true; if (baseBlock is Block) { var block = (Block)baseBlock; foreach (var block2 in block.GetTargets()) AddBaseBlock(block2); } else if (baseBlock is ScopeBlock) { var scopeBlock = (ScopeBlock)baseBlock; AddScopeBlock(scopeBlock); if (scopeBlock.BaseBlocks != null && scopeBlock.BaseBlocks.Count > 0) AddBaseBlock(scopeBlock.BaseBlocks[0]); } else throw new ApplicationException(string.Format("Unknown BaseBlock type {0}", baseBlock.GetType())); } // Add a block to be processed later, including all its enclosing ScopeBlocks. void AddBaseBlock(BaseBlock baseBlock) { for (BaseBlock bb = baseBlock; bb != null; bb = bb.Parent) baseBlocksToCheck.Push(bb); } void ProcessScopeBlock(ScopeBlock scopeBlock) { if (scopeBlock == null || checkedScopeBlocks.ContainsKey(scopeBlock)) return; checkedScopeBlocks[scopeBlock] = true; AddBaseBlock(scopeBlock); if (scopeBlock is TryBlock) { var tryBlock = (TryBlock)scopeBlock; foreach (var handler in tryBlock.TryHandlerBlocks) AddScopeBlock(handler); } else if (scopeBlock is TryHandlerBlock) { var tryHandlerBlock = (TryHandlerBlock)scopeBlock; AddScopeBlock(tryHandlerBlock.FilterHandlerBlock); AddScopeBlock(tryHandlerBlock.HandlerBlock); } } } }