/CoreServices/FuncName.cs

# · C# · 270 lines · 218 code · 28 blank · 24 comment · 60 complexity · 85a1f7b8b657295ed719101f7a696196 MD5 · raw file

  1. // Copyright (c) 2009, Semyon A. Chertkov (semyonc@gmail.com)
  2. // All rights reserved.
  3. //
  4. // Redistribution and use in source and binary forms, with or without
  5. // modification, are permitted provided that the following conditions are met:
  6. // * Redistributions of source code must retain the above copyright
  7. // notice, this list of conditions and the following disclaimer.
  8. // * Redistributions in binary form must reproduce the above copyright
  9. // notice, this list of conditions and the following disclaimer in the
  10. // documentation and/or other materials provided with the distribution.
  11. // * Neither the name of author nor the
  12. // names of its contributors may be used to endorse or promote products
  13. // derived from this software without specific prior written permission.
  14. //
  15. // THIS SOFTWARE IS PROVIDED ''AS IS'' AND ANY
  16. // EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
  17. // WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
  18. // DISCLAIMED. IN NO EVENT SHALL AUTHOR BE LIABLE FOR ANY
  19. // DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
  20. // (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
  21. // LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
  22. // ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
  23. // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
  24. // SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  25. using System;
  26. using System.Collections.Generic;
  27. using System.Text;
  28. namespace DataEngine.CoreServices
  29. {
  30. public class FuncName
  31. {
  32. public Object ID { get; private set; }
  33. public Type[] Signature { get; private set; }
  34. public bool VariableLength { get; private set; }
  35. public int Arity
  36. {
  37. get
  38. {
  39. return Signature.Length;
  40. }
  41. }
  42. public Type VariableParamType
  43. {
  44. get
  45. {
  46. if (!VariableLength)
  47. return null;
  48. return Signature[Arity - 1];
  49. }
  50. }
  51. public FuncName(Object id)
  52. {
  53. ID = id;
  54. Signature = new Type[0];
  55. VariableLength = false;
  56. }
  57. public FuncName(Object id, Type[] signature)
  58. {
  59. ID = id;
  60. Signature = signature;
  61. VariableLength = false;
  62. }
  63. public FuncName(Object id, Executive.Parameter[] parameters)
  64. {
  65. ID = id;
  66. int len = parameters.Length;
  67. Signature = new Type[len];
  68. for (int k = 0; k < len; k++)
  69. Signature[k] = parameters[k].Type;
  70. if (len > 0 && parameters[len - 1].VariableParam)
  71. VariableLength = true;
  72. }
  73. private void FormatTypeName(StringBuilder result, Type type)
  74. {
  75. if (type.IsGenericType)
  76. {
  77. string genericName = type.GetGenericTypeDefinition().FullName.Replace('+', '.');
  78. int tickIndex = genericName.IndexOf('`');
  79. result.Append(tickIndex != -1 ? genericName.Substring(0, tickIndex) : genericName);
  80. FormatTypeArgs(result, type.GetGenericArguments());
  81. }
  82. else if (type.IsGenericParameter)
  83. {
  84. result.Append(type.Name);
  85. }
  86. else
  87. {
  88. result.Append(type.FullName.Replace('+', '.'));
  89. }
  90. }
  91. private void FormatTypeArgs(StringBuilder result, Type[] types)
  92. {
  93. if (types.Length > 0)
  94. {
  95. result.Append("<");
  96. for (int i = 0; i < types.Length; i++)
  97. {
  98. if (i > 0) result.Append(", ");
  99. FormatTypeName(result, types[i]);
  100. }
  101. result.Append(">");
  102. }
  103. }
  104. public override string ToString()
  105. {
  106. StringBuilder sb = new StringBuilder();
  107. if (Arity != -1)
  108. {
  109. sb.AppendFormat("{0}/{1}(", ID, Arity);
  110. FormatTypeArgs(sb, Signature);
  111. sb.Append(")");
  112. }
  113. return sb.ToString();
  114. }
  115. public override int GetHashCode()
  116. {
  117. int hashCode = ID.GetHashCode();
  118. foreach (Type type in Signature)
  119. hashCode = hashCode * 37 + type.GetHashCode();
  120. return hashCode;
  121. }
  122. public override bool Equals(object obj)
  123. {
  124. if (obj is FuncName)
  125. return Equals((FuncName)obj);
  126. else
  127. return base.Equals(obj);
  128. }
  129. public bool Equals(FuncName name)
  130. {
  131. if (name.ID == ID && name.Arity == Arity &&
  132. name.VariableLength == VariableLength)
  133. {
  134. for (int k = 0; k < Signature.Length; k++)
  135. if (Signature[k] != name.Signature[k])
  136. return false;
  137. return true;
  138. }
  139. else
  140. return false;
  141. }
  142. public Type GetParameterType(int index)
  143. {
  144. if (!VariableLength || index < Arity - 1)
  145. return Signature[index];
  146. else
  147. return Signature[Arity - 1];
  148. }
  149. public bool Match(FuncName name, bool anyType)
  150. {
  151. if (name.ID == ID)
  152. {
  153. if (name.Arity == Arity || (VariableLength && Arity < name.Arity))
  154. {
  155. for (int k = 0; k < Signature.Length; k++)
  156. if (anyType)
  157. {
  158. if (!Signature[k].IsAssignableFrom(name.Signature[k]))
  159. return false;
  160. }
  161. else
  162. {
  163. if (Signature[k] != name.Signature[k])
  164. return false;
  165. }
  166. if (VariableLength && Arity < name.Arity)
  167. {
  168. int last = Signature.Length - 1;
  169. for (int k = Signature.Length; k < name.Arity; k++)
  170. if (anyType)
  171. {
  172. if (!Signature[last].IsAssignableFrom(name.Signature[k]))
  173. return false;
  174. }
  175. else
  176. {
  177. if (Signature[last] != name.Signature[k])
  178. return false;
  179. }
  180. }
  181. return true;
  182. }
  183. if (VariableLength && Arity == 1 && name.Arity == 0)
  184. return true;
  185. }
  186. return false;
  187. }
  188. }
  189. public class MacroFuncName
  190. {
  191. public object ID { get; private set; }
  192. public int Arity { get; private set; }
  193. public bool VariableLength { get; private set; }
  194. public MacroFuncName(Object id, int arity)
  195. {
  196. ID = id;
  197. Arity = arity;
  198. VariableLength = false;
  199. }
  200. public MacroFuncName(Object id)
  201. {
  202. ID = id;
  203. Arity = -1;
  204. VariableLength = true;
  205. }
  206. public override string ToString()
  207. {
  208. StringBuilder sb = new StringBuilder();
  209. if (Arity != -1)
  210. {
  211. if (VariableLength)
  212. sb.Append(ID);
  213. else
  214. sb.AppendFormat("{0}/{1}", ID, Arity);
  215. sb.Append(")");
  216. }
  217. return sb.ToString();
  218. }
  219. public override bool Equals(object obj)
  220. {
  221. if (obj is MacroFuncName)
  222. return Equals((MacroFuncName)obj);
  223. else
  224. return base.Equals(obj);
  225. }
  226. public bool Equals(MacroFuncName name)
  227. {
  228. if (name.ID == ID && name.Arity == Arity)
  229. return true;
  230. else
  231. return false;
  232. }
  233. public override int GetHashCode()
  234. {
  235. return ID.GetHashCode() * 37 + Arity.GetHashCode();
  236. }
  237. public bool Match(MacroFuncName name)
  238. {
  239. return ID == name.ID && (VariableLength || Arity == name.Arity);
  240. }
  241. }
  242. }