/CSharpFactoryVS2010/CSharpFactory/ProjectModel/Statements/BlockStatement.cs

# · C# · 260 lines · 136 code · 32 blank · 92 comment · 16 complexity · 5f739061386ff1f413ea133b5e490af8 MD5 · raw file

  1. using System.Collections.Generic;
  2. using CSharpFactory.ParserFiles;
  3. using CSharpFactory.Semantics;
  4. namespace CSharpFactory.ProjectModel
  5. {
  6. // ==================================================================================
  7. /// <summary>
  8. /// This type represents a block of statements.
  9. /// </summary>
  10. // ==================================================================================
  11. public class BlockStatement : Statement, IBlockOwner
  12. {
  13. #region Private fields
  14. private readonly StatementCollection _Statements;
  15. private readonly VariableCollection _Variables = new VariableCollection();
  16. private readonly List<IBlockOwner> _ChildBlocks = new List<IBlockOwner>();
  17. #endregion
  18. #region Lifecyle methods
  19. // --------------------------------------------------------------------------------
  20. /// <summary>
  21. /// Creates a new block statement declaration.
  22. /// </summary>
  23. /// <param name="token">Token providing position information.</param>
  24. /// <param name="parser">Parser instance creating this element.</param>
  25. /// <param name="parentBlock">Parent block of this block.</param>
  26. // --------------------------------------------------------------------------------
  27. public BlockStatement(Token token, CSharpSyntaxParser parser, IBlockOwner parentBlock)
  28. : base(token, parser, parentBlock)
  29. {
  30. _Statements = new StatementCollection(this);
  31. }
  32. #endregion
  33. #region Public properties
  34. // --------------------------------------------------------------------------------
  35. /// <summary>
  36. /// Gets the list of statements in this block.
  37. /// </summary>
  38. // --------------------------------------------------------------------------------
  39. public StatementCollection Statements
  40. {
  41. get { return _Statements; }
  42. }
  43. #endregion
  44. #region IBlockOwner Members
  45. // --------------------------------------------------------------------------------
  46. /// <summary>
  47. /// Gets the element owning the block;
  48. /// </summary>
  49. // --------------------------------------------------------------------------------
  50. public IBlockOwner Owner
  51. {
  52. get { return this; }
  53. }
  54. // --------------------------------------------------------------------------------
  55. /// <summary>
  56. /// Adds a new statement to the method body.
  57. /// </summary>
  58. /// <param name="statement">Statement to add.</param>
  59. // --------------------------------------------------------------------------------
  60. public void Add(Statement statement)
  61. {
  62. Statements.Add(statement);
  63. statement.SetParent(this);
  64. IBlockOwner blockStatement = statement as IBlockOwner;
  65. if (blockStatement != null)
  66. {
  67. _ChildBlocks.Add(blockStatement);
  68. }
  69. }
  70. // --------------------------------------------------------------------------------
  71. /// <summary>
  72. /// Gets the list ob child block in this one.
  73. /// </summary>
  74. // --------------------------------------------------------------------------------
  75. public List<IBlockOwner> ChildBlocks
  76. {
  77. get { return _ChildBlocks; }
  78. }
  79. // --------------------------------------------------------------------------------
  80. /// <summary>
  81. /// Gets the collection of variables belonging to this block.
  82. /// </summary>
  83. // --------------------------------------------------------------------------------
  84. public VariableCollection Variables
  85. {
  86. get { return _Variables; }
  87. }
  88. // --------------------------------------------------------------------------------
  89. /// <summary>
  90. /// Adds a new localVariable the block.
  91. /// </summary>
  92. /// <param name="localVariable">Variable to add to the block.</param>
  93. // --------------------------------------------------------------------------------
  94. public virtual void Add(LocalVariable localVariable)
  95. {
  96. AddVariableToBlock(this, localVariable);
  97. }
  98. #endregion
  99. #region Static helper methods
  100. // --------------------------------------------------------------------------------
  101. /// <summary>
  102. /// Adds a new local variable to the specified block.
  103. /// </summary>
  104. /// <param name="block">Block that owns the local variable.</param>
  105. /// <param name="localVariable">Variable to add to the block.</param>
  106. // --------------------------------------------------------------------------------
  107. public static void AddVariableToBlock(IBlockOwner block, LocalVariable localVariable)
  108. {
  109. // --- Check for duplication in the same scope
  110. if (block.Variables.Contains(localVariable))
  111. {
  112. localVariable.Parser.Error0128(localVariable);
  113. return;
  114. }
  115. // --- Cycle from this block to its parents
  116. IBlockOwner currentBlock = block;
  117. do
  118. {
  119. if (currentBlock.ParentBlock == null)
  120. {
  121. // --- This block does not have a parent, maybe its parent statement has...
  122. Statement stmt = currentBlock.Owner as Statement;
  123. currentBlock = stmt == null ? null : stmt.Parent as IBlockOwner;
  124. }
  125. else currentBlock = currentBlock.ParentBlock;
  126. // --- Leave the block, if there is no parent
  127. if (currentBlock == null) break;
  128. // --- Check the parent block
  129. if (currentBlock.Variables.Contains(localVariable))
  130. {
  131. localVariable.Parser.Error0136(localVariable, "parent");
  132. return;
  133. }
  134. } while (true);
  135. // --- Now check embedded blocks recursively
  136. if (IsVariableInChildBlock(block, localVariable)) return;
  137. // --- Ok, this is a new variable.
  138. block.Variables.Add(localVariable);
  139. }
  140. // --------------------------------------------------------------------------------
  141. /// <summary>
  142. /// Checks if the specified local variable is defined in any of the childs of the
  143. /// given block
  144. /// </summary>
  145. /// <param name="block">Block to check</param>
  146. /// <param name="localVariable">Local variable to check</param>
  147. /// <returns>
  148. /// True, if any of the child blocks defines the variable.; otherwise, false.
  149. /// </returns>
  150. // --------------------------------------------------------------------------------
  151. private static bool IsVariableInChildBlock(IBlockOwner block, LocalVariable localVariable)
  152. {
  153. foreach (IBlockOwner childBlock in block.ChildBlocks)
  154. {
  155. if (childBlock.Variables.Contains(localVariable))
  156. {
  157. localVariable.Parser.Error0136(localVariable, "child");
  158. return true;
  159. }
  160. if (IsVariableInChildBlock(childBlock, localVariable)) return true;
  161. }
  162. return false;
  163. }
  164. #endregion
  165. #region IUsesResolutionContext implementation
  166. // --------------------------------------------------------------------------------
  167. /// <summary>
  168. /// Resolves all unresolved type references.
  169. /// </summary>
  170. /// <param name="contextType">Type of resolution context.</param>
  171. /// <param name="declarationScope">Current type declaration context.</param>
  172. /// <param name="parameterScope">Current type parameter declaration scope.</param>
  173. // --------------------------------------------------------------------------------
  174. public override void ResolveTypeReferences(ResolutionContext contextType,
  175. ITypeDeclarationScope declarationScope,
  176. ITypeParameterScope parameterScope)
  177. {
  178. base.ResolveTypeReferences(contextType, declarationScope, parameterScope);
  179. ResolveTypeReferences(this, contextType, declarationScope, parameterScope);
  180. }
  181. #endregion
  182. #region Iterator methods
  183. // --------------------------------------------------------------------------------
  184. /// <summary>
  185. /// Iterator enumerating all nested statements belonging to this one.
  186. /// </summary>
  187. /// <value>Returns recursively all nested statements.</value>
  188. /// <remarks>Statements are returned in the order of their declaration.</remarks>
  189. // --------------------------------------------------------------------------------
  190. public override IEnumerable<Statement> NestedStatements
  191. {
  192. get
  193. {
  194. foreach (Statement stmt in _Statements)
  195. {
  196. BlockStatement block = stmt as BlockStatement;
  197. if (block == null)
  198. {
  199. yield return stmt;
  200. }
  201. else
  202. {
  203. foreach (Statement nested in block.NestedStatements)
  204. yield return nested;
  205. }
  206. }
  207. }
  208. }
  209. // --------------------------------------------------------------------------------
  210. /// <summary>
  211. /// Iterator enumerating all direclty nested statements belonging to this one.
  212. /// </summary>
  213. /// <value>Returns only the directly nested statements, does not do recursion.</value>
  214. /// <remarks>Statements are returned in the order of their declaration.</remarks>
  215. // --------------------------------------------------------------------------------
  216. public override IEnumerable<Statement> DirectNestedStatements
  217. {
  218. get
  219. {
  220. foreach (Statement stmt in _Statements)
  221. {
  222. yield return stmt;
  223. }
  224. }
  225. }
  226. #endregion
  227. }
  228. }