/Release/v2.0.0.0/StaMa_State_Machine_Controller_Library/StaMa/Region.cs

# · C# · 397 lines · 227 code · 56 blank · 114 comment · 18 complexity · 0e158555c073497c78c599546edd4269 MD5 · raw file

  1. #region Region.cs file
  2. //
  3. // StaMa state machine controller library
  4. //
  5. // Copyright (c) 2005, Roland Schneider. All rights reserved.
  6. //
  7. #endregion
  8. using System;
  9. using System.Collections.Generic;
  10. using System.Diagnostics;
  11. namespace StaMa
  12. {
  13. /// <summary>
  14. /// Represents a container for states and defines the initial <see cref="State"/> for the contained states.
  15. /// </summary>
  16. /// <remarks>
  17. /// Instances of this class will be created through the <see cref="StateMachineTemplate.Region"/> method.
  18. /// </remarks>
  19. [DebuggerDisplay("DistinguishedName={DistinguishedName}")]
  20. public class Region
  21. {
  22. private State m_parentState;
  23. private StateCollection m_states;
  24. private State m_initialState;
  25. private string m_initialStateName;
  26. private uint m_historyIndex;
  27. private uint m_stateConfigurationIndex;
  28. /// <summary>
  29. /// Initializes a new <see cref="Region"/> instance.
  30. /// </summary>
  31. /// <param name="parentState">
  32. /// The <see cref="State"/> instance that contains this <see cref="Region"/>. Shall be <c>null</c> for the root <see cref="Region"/> instance.
  33. /// </param>
  34. /// <param name="initialStateName">
  35. /// The name of the initial <see cref="State"/> of the <see cref="Region"/>.
  36. /// </param>
  37. /// <param name="historyIndex">
  38. /// If the <see cref="Region"/> shall have a history, the index of the slot where
  39. /// this <see cref="Region"/> stores its history in the <see cref="StateMachine"/> instance.
  40. /// If the region has no history, <see cref="uint.MaxValue">uint.MaxValue</see>.
  41. /// </param>
  42. /// <param name="stateMachineTemplate">
  43. /// The embedding <see cref="StateMachineTemplate"/> instance.
  44. /// Internally used for creating the <see cref="States"/> collection.
  45. /// </param>
  46. /// <remarks>
  47. /// Instances of this class will be created through the <see cref="StateMachineTemplate.Region"/> method.
  48. /// </remarks>
  49. internal protected Region(State parentState, string initialStateName, uint historyIndex, StateMachineTemplate stateMachineTemplate)
  50. {
  51. m_parentState = parentState;
  52. m_states = stateMachineTemplate.InternalCreateStateCollection();
  53. m_initialState = null;
  54. m_initialStateName = initialStateName;
  55. m_historyIndex = historyIndex;
  56. m_stateConfigurationIndex = uint.MaxValue;
  57. }
  58. internal void EndRegion()
  59. {
  60. foreach (State state in m_states)
  61. {
  62. if (state.Name == m_initialStateName)
  63. {
  64. m_initialState = state;
  65. break;
  66. }
  67. }
  68. if (m_initialState == null)
  69. {
  70. throw new InvalidOperationException("The initialStateName of the \"Region()\" statement doesn't correspond with an embedded State.");
  71. }
  72. }
  73. /// <summary>
  74. /// Overrides <see cref="System.Object.ToString"/> and returns the execution order of the <see cref="Region"/>.
  75. /// </summary>
  76. public override string ToString()
  77. {
  78. return this.ExecutionOrder.ToString();
  79. }
  80. internal string DistinguishedName
  81. {
  82. get
  83. {
  84. if (m_parentState != null)
  85. {
  86. return m_parentState.DistinguishedName + '~' + this.ExecutionOrder.ToString();
  87. }
  88. else
  89. {
  90. return '~' + this.ExecutionOrder.ToString();
  91. }
  92. }
  93. }
  94. /// <summary>
  95. /// Gets the <see cref="State"/> that aggregates this <see cref="Region"/>.
  96. /// </summary>
  97. public State Parent
  98. {
  99. get
  100. {
  101. return m_parentState;
  102. }
  103. }
  104. /// <summary>
  105. /// Gets the list of <see cref="State"/>s of this <see cref="Region"/>.
  106. /// </summary>
  107. /// <value>
  108. /// A <see cref="StateCollection"/> instance. At least one item (namely the
  109. /// <see cref="InitialState"/>) will be present in this collection.
  110. /// </value>
  111. public StateCollection States
  112. {
  113. get
  114. {
  115. return m_states;
  116. }
  117. }
  118. /// <summary>
  119. /// The sequential index of this <see cref="Region"/> within
  120. /// the <see cref="State.Regions"/> collection of the parent <see cref="State"/>.
  121. /// </summary>
  122. /// <value>
  123. /// A <see cref="uint"/> greater or equal 1.
  124. /// </value>
  125. public uint ExecutionOrder
  126. {
  127. get
  128. {
  129. if (m_parentState == null)
  130. {
  131. return 1;
  132. }
  133. else
  134. {
  135. #warning Make Regions inherit from LinkedList and follow Prev until we reach First.
  136. uint index = 1;
  137. foreach (Region region in this.Parent.Regions)
  138. {
  139. if (region == this)
  140. {
  141. return index;
  142. }
  143. index++;
  144. }
  145. // Logical error
  146. throw new Exception("Internal error: Region instance not contained in regions list of parent state.");
  147. }
  148. }
  149. }
  150. /// <summary>
  151. /// The <see cref="State"/> that will be made active when the <see cref="Region"/> is entered
  152. /// the first time.
  153. /// </summary>
  154. public State InitialState
  155. {
  156. get
  157. {
  158. return m_initialState;
  159. }
  160. }
  161. /// <summary>
  162. /// The slot within a <see cref="StateConfiguration"/> that will be used by
  163. /// the active state of this <see cref="Region"/>.
  164. /// </summary>
  165. public uint StateConfigurationIndex
  166. {
  167. get
  168. {
  169. return m_stateConfigurationIndex;
  170. }
  171. }
  172. /// <summary>
  173. /// Indicates whether the recently active <see cref="State"/> will be made
  174. /// active on rentry of the <see cref="Region"/>.
  175. /// </summary>
  176. /// <value>
  177. /// <c>true</c> if the recently active <see cref="State"/> will be made
  178. /// active on rentry; <c>false</c> if the <see cref="InitialState"/> will be made active.
  179. /// </value>
  180. public bool HasHistory
  181. {
  182. get
  183. {
  184. return (m_historyIndex != uint.MaxValue);
  185. }
  186. }
  187. /// <summary>
  188. /// Gets the index of the slot where this <see cref="Region"/> stores its
  189. /// history in the <see cref="StateMachine"/> instance.
  190. /// </summary>
  191. /// <value>
  192. /// A <see cref="uint"/> if the <see cref="Region"/> has a history; otherwise, <see cref="uint.MaxValue">uint.MaxValue</see>.
  193. /// </value>
  194. public uint HistoryIndex
  195. {
  196. get
  197. {
  198. return m_historyIndex;
  199. }
  200. }
  201. internal bool PassThrough(IStateMachineTemplateVisitor visitor)
  202. {
  203. bool continuePassThrough = visitor.Region(this);
  204. if (!continuePassThrough)
  205. {
  206. return false;
  207. }
  208. // Iterate through states.
  209. foreach (State state in this.States)
  210. {
  211. continuePassThrough = state.PassThrough(visitor);
  212. if (!continuePassThrough)
  213. {
  214. return false;
  215. }
  216. }
  217. continuePassThrough = visitor.EndRegion(this);
  218. if (!continuePassThrough)
  219. {
  220. return false;
  221. }
  222. return true;
  223. }
  224. internal void AddState(State state)
  225. {
  226. m_states.Add(state);
  227. }
  228. internal void FixupAndCalcStateConfigMetrics(ref uint stateConfigurationIndex, out uint stateConfigMax, out uint stateBasConfigMax)
  229. {
  230. // Set the slot for the active state reference.
  231. // For every region we have to reserve one slot in
  232. // the active state array.
  233. m_stateConfigurationIndex = stateConfigurationIndex;
  234. stateConfigurationIndex = stateConfigurationIndex + 1; // Increase value (children and siblings will start on a new index)
  235. uint stateConfigLocalMax = 0;
  236. stateBasConfigMax = 0; // Return the maximum of concurrent transitions for all substates
  237. foreach (State state in this.States)
  238. {
  239. uint stateConfigSize = 0; // Size of pointers needed for this state and its offspring in a state config array
  240. uint stateBasConfigSize = 0; // Size of pointers needed for this state in a base state configuration array
  241. state.FixupAndCalcStateConfigMetrics(stateConfigurationIndex, out stateConfigSize, out stateBasConfigSize);
  242. if (stateConfigSize > stateConfigLocalMax) stateConfigLocalMax = stateConfigSize; // Find the maximum needed over all states in this region
  243. if (stateBasConfigSize > stateBasConfigMax) stateBasConfigMax = stateBasConfigSize;
  244. }
  245. stateConfigMax = stateConfigLocalMax + 1; // Return the maximum of the offspring plus one for ourselves
  246. stateConfigurationIndex += stateConfigLocalMax; // Shift the index to reserve space for this region's offspring
  247. }
  248. internal void FinalFixup(StateMachineTemplate stateMachineTemplate)
  249. {
  250. // Recurse to sub items
  251. foreach (State state in this.States)
  252. {
  253. foreach (Transition transition in state.Transitions)
  254. {
  255. transition.FinalFixup(stateMachineTemplate);
  256. }
  257. foreach (Region subRegion in state.Regions)
  258. {
  259. subRegion.FinalFixup(stateMachineTemplate);
  260. }
  261. }
  262. }
  263. }
  264. /// <summary>
  265. /// Represents a list of <see cref="State"/> instances.
  266. /// </summary>
  267. public class StateCollection
  268. {
  269. private List<State> m_states;
  270. /// <summary>
  271. /// Initializes a new <see cref="StateCollection"/> instance.
  272. /// </summary>
  273. internal protected StateCollection()
  274. {
  275. m_states = new List<State>();
  276. }
  277. /// <summary>
  278. /// Returns an enumerator that iterates through the <see cref="StateCollection"/>.
  279. /// </summary>
  280. /// <returns>
  281. /// A <see cref="System.Collections.IEnumerator"/> for the <see cref="StateCollection"/>.
  282. /// </returns>
  283. public System.Collections.IEnumerator GetEnumerator()
  284. {
  285. return ((System.Collections.IEnumerable)m_states).GetEnumerator();
  286. }
  287. /// <summary>
  288. /// Gets the <see cref="State"/> with a specific name within the <see cref="Region"/>.
  289. /// </summary>
  290. /// <param name="stateName">
  291. /// The name of the <see cref="State"/>.
  292. /// </param>
  293. /// <returns>
  294. /// The <see cref="State"/> with the specified name, or <c>null</c> if there is no such <see cref="State"/>.
  295. /// </returns>
  296. public State this[string stateName]
  297. {
  298. get
  299. {
  300. foreach (State state in m_states)
  301. {
  302. if (stateName == state.Name)
  303. {
  304. return state;
  305. }
  306. }
  307. return null;
  308. }
  309. }
  310. /// <summary>
  311. /// Gets the <see cref="State"/> at the specified index.
  312. /// </summary>
  313. /// <param name="index">
  314. /// The zero-based index of the <see cref="State"/> to get.
  315. /// </param>
  316. /// <returns>
  317. /// The <see cref="State"/> at the index.
  318. /// </returns>
  319. public State this[int index]
  320. {
  321. get
  322. {
  323. return m_states[index];
  324. }
  325. }
  326. /// <summary>
  327. /// Gets the number of elements contained in the <see cref="StateCollection"/>.
  328. /// </summary>
  329. public int Count
  330. {
  331. get
  332. {
  333. return m_states.Count;
  334. }
  335. }
  336. internal void Add(State state)
  337. {
  338. m_states.Add(state);
  339. }
  340. }
  341. }