Funcletizer.cs 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445
  1. using System;
  2. using System.Collections.Generic;
  3. using System.Collections.ObjectModel;
  4. using System.Linq;
  5. using System.Linq.Expressions;
  6. namespace System.Data.Linq.SqlClient {
  7. using System.Data.Linq.Mapping;
  8. using System.Diagnostics.CodeAnalysis;
  9. internal static class Funcletizer {
  10. internal static Expression Funcletize(Expression expression) {
  11. return new Localizer(new LocalMapper().MapLocals(expression)).Localize(expression);
  12. }
  13. class Localizer : ExpressionVisitor {
  14. Dictionary<Expression, bool> locals;
  15. internal Localizer(Dictionary<Expression, bool> locals) {
  16. this.locals = locals;
  17. }
  18. internal Expression Localize(Expression expression) {
  19. return this.Visit(expression);
  20. }
  21. internal override Expression Visit(Expression exp) {
  22. if (exp == null) {
  23. return null;
  24. }
  25. if (this.locals.ContainsKey(exp)) {
  26. return MakeLocal(exp);
  27. }
  28. if (exp.NodeType == (ExpressionType)InternalExpressionType.Known) {
  29. return exp;
  30. }
  31. return base.Visit(exp);
  32. }
  33. private static Expression MakeLocal(Expression e) {
  34. if (e.NodeType == ExpressionType.Constant) {
  35. return e;
  36. }
  37. else if (e.NodeType == ExpressionType.Convert || e.NodeType == ExpressionType.ConvertChecked) {
  38. UnaryExpression ue = (UnaryExpression)e;
  39. if (ue.Type == typeof(object)) {
  40. Expression local = MakeLocal(ue.Operand);
  41. return (e.NodeType == ExpressionType.Convert) ? Expression.Convert(local, e.Type) : Expression.ConvertChecked(local, e.Type);
  42. }
  43. // convert a const null
  44. if (ue.Operand.NodeType == ExpressionType.Constant) {
  45. ConstantExpression c = (ConstantExpression)ue.Operand;
  46. if (c.Value == null) {
  47. return Expression.Constant(null, ue.Type);
  48. }
  49. }
  50. }
  51. return Expression.Invoke(Expression.Constant(Expression.Lambda(e).Compile()));
  52. }
  53. }
  54. class DependenceChecker : ExpressionVisitor {
  55. HashSet<ParameterExpression> inScope = new HashSet<ParameterExpression>();
  56. bool isIndependent = true;
  57. /// <summary>
  58. /// This method returns 'true' when the expression doesn't reference any parameters
  59. /// from outside the scope of the expression.
  60. /// </summary>
  61. static public bool IsIndependent(Expression expression) {
  62. var v = new DependenceChecker();
  63. v.Visit(expression);
  64. return v.isIndependent;
  65. }
  66. internal override Expression VisitLambda(LambdaExpression lambda) {
  67. foreach (var p in lambda.Parameters) {
  68. this.inScope.Add(p);
  69. }
  70. return base.VisitLambda(lambda);
  71. }
  72. internal override Expression VisitParameter(ParameterExpression p) {
  73. this.isIndependent &= this.inScope.Contains(p);
  74. return p;
  75. }
  76. }
  77. class LocalMapper : ExpressionVisitor {
  78. bool isRemote;
  79. Dictionary<Expression, bool> locals;
  80. internal Dictionary<Expression, bool> MapLocals(Expression expression) {
  81. this.locals = new Dictionary<Expression, bool>();
  82. this.isRemote = false;
  83. this.Visit(expression);
  84. return this.locals;
  85. }
  86. internal override Expression Visit(Expression expression) {
  87. if (expression == null) {
  88. return null;
  89. }
  90. bool saveIsRemote = this.isRemote;
  91. switch (expression.NodeType) {
  92. case (ExpressionType)InternalExpressionType.Known:
  93. return expression;
  94. case (ExpressionType)ExpressionType.Constant:
  95. break;
  96. default:
  97. this.isRemote = false;
  98. base.Visit(expression);
  99. if (!this.isRemote
  100. && expression.NodeType != ExpressionType.Lambda
  101. && expression.NodeType != ExpressionType.Quote
  102. && DependenceChecker.IsIndependent(expression)) {
  103. this.locals[expression] = true; // Not 'Add' because the same expression may exist in the tree twice.
  104. }
  105. break;
  106. }
  107. if (typeof(ITable).IsAssignableFrom(expression.Type) ||
  108. typeof(DataContext).IsAssignableFrom(expression.Type)) {
  109. this.isRemote = true;
  110. }
  111. this.isRemote |= saveIsRemote;
  112. return expression;
  113. }
  114. internal override Expression VisitMemberAccess(MemberExpression m) {
  115. base.VisitMemberAccess(m);
  116. this.isRemote |= (m.Expression != null && typeof(ITable).IsAssignableFrom(m.Expression.Type));
  117. return m;
  118. }
  119. internal override Expression VisitMethodCall(MethodCallExpression m) {
  120. base.VisitMethodCall(m);
  121. this.isRemote |= m.Method.DeclaringType == typeof(System.Data.Linq.Provider.DataManipulation)
  122. || Attribute.IsDefined(m.Method, typeof(FunctionAttribute));
  123. return m;
  124. }
  125. }
  126. }
  127. internal abstract class ExpressionVisitor {
  128. internal ExpressionVisitor() {
  129. }
  130. [SuppressMessage("Microsoft.Maintainability", "CA1502:AvoidExcessiveComplexity", Justification = "These issues are related to our use of if-then and case statements for node types, which adds to the complexity count however when reviewed they are easy to navigate and understand.")]
  131. [SuppressMessage("Microsoft.Performance", "CA1800:DoNotCastUnnecessarily", Justification = "[....]: Cast is dependent on node type and casts do not happen unecessarily in a single code path.")]
  132. internal virtual Expression Visit(Expression exp) {
  133. if (exp == null)
  134. return exp;
  135. switch (exp.NodeType) {
  136. case ExpressionType.Negate:
  137. case ExpressionType.NegateChecked:
  138. case ExpressionType.Not:
  139. case ExpressionType.Convert:
  140. case ExpressionType.ConvertChecked:
  141. case ExpressionType.ArrayLength:
  142. case ExpressionType.Quote:
  143. case ExpressionType.TypeAs:
  144. return this.VisitUnary((UnaryExpression)exp);
  145. case ExpressionType.Add:
  146. case ExpressionType.AddChecked:
  147. case ExpressionType.Subtract:
  148. case ExpressionType.SubtractChecked:
  149. case ExpressionType.Multiply:
  150. case ExpressionType.MultiplyChecked:
  151. case ExpressionType.Divide:
  152. case ExpressionType.Modulo:
  153. case ExpressionType.Power:
  154. case ExpressionType.And:
  155. case ExpressionType.AndAlso:
  156. case ExpressionType.Or:
  157. case ExpressionType.OrElse:
  158. case ExpressionType.LessThan:
  159. case ExpressionType.LessThanOrEqual:
  160. case ExpressionType.GreaterThan:
  161. case ExpressionType.GreaterThanOrEqual:
  162. case ExpressionType.Equal:
  163. case ExpressionType.NotEqual:
  164. case ExpressionType.Coalesce:
  165. case ExpressionType.ArrayIndex:
  166. case ExpressionType.RightShift:
  167. case ExpressionType.LeftShift:
  168. case ExpressionType.ExclusiveOr:
  169. return this.VisitBinary((BinaryExpression)exp);
  170. case ExpressionType.TypeIs:
  171. return this.VisitTypeIs((TypeBinaryExpression)exp);
  172. case ExpressionType.Conditional:
  173. return this.VisitConditional((ConditionalExpression)exp);
  174. case ExpressionType.Constant:
  175. return this.VisitConstant((ConstantExpression)exp);
  176. case ExpressionType.Parameter:
  177. return this.VisitParameter((ParameterExpression)exp);
  178. case ExpressionType.MemberAccess:
  179. return this.VisitMemberAccess((MemberExpression)exp);
  180. case ExpressionType.Call:
  181. return this.VisitMethodCall((MethodCallExpression)exp);
  182. case ExpressionType.Lambda:
  183. return this.VisitLambda((LambdaExpression)exp);
  184. case ExpressionType.New:
  185. return this.VisitNew((NewExpression)exp);
  186. case ExpressionType.NewArrayInit:
  187. case ExpressionType.NewArrayBounds:
  188. return this.VisitNewArray((NewArrayExpression)exp);
  189. case ExpressionType.Invoke:
  190. return this.VisitInvocation((InvocationExpression)exp);
  191. case ExpressionType.MemberInit:
  192. return this.VisitMemberInit((MemberInitExpression)exp);
  193. case ExpressionType.ListInit:
  194. return this.VisitListInit((ListInitExpression)exp);
  195. case ExpressionType.UnaryPlus:
  196. if (exp.Type == typeof(TimeSpan))
  197. return this.VisitUnary((UnaryExpression)exp);
  198. throw Error.UnhandledExpressionType(exp.NodeType);
  199. default:
  200. throw Error.UnhandledExpressionType(exp.NodeType);
  201. }
  202. }
  203. internal virtual MemberBinding VisitBinding(MemberBinding binding) {
  204. switch (binding.BindingType) {
  205. case MemberBindingType.Assignment:
  206. return this.VisitMemberAssignment((MemberAssignment)binding);
  207. case MemberBindingType.MemberBinding:
  208. return this.VisitMemberMemberBinding((MemberMemberBinding)binding);
  209. case MemberBindingType.ListBinding:
  210. return this.VisitMemberListBinding((MemberListBinding)binding);
  211. default:
  212. throw Error.UnhandledBindingType(binding.BindingType);
  213. }
  214. }
  215. internal virtual ElementInit VisitElementInitializer(ElementInit initializer) {
  216. ReadOnlyCollection<Expression> arguments = this.VisitExpressionList(initializer.Arguments);
  217. if (arguments != initializer.Arguments) {
  218. return Expression.ElementInit(initializer.AddMethod, arguments);
  219. }
  220. return initializer;
  221. }
  222. internal virtual Expression VisitUnary(UnaryExpression u) {
  223. Expression operand = this.Visit(u.Operand);
  224. if (operand != u.Operand) {
  225. return Expression.MakeUnary(u.NodeType, operand, u.Type, u.Method);
  226. }
  227. return u;
  228. }
  229. internal virtual Expression VisitBinary(BinaryExpression b) {
  230. Expression left = this.Visit(b.Left);
  231. Expression right = this.Visit(b.Right);
  232. if (left != b.Left || right != b.Right) {
  233. return Expression.MakeBinary(b.NodeType, left, right, b.IsLiftedToNull, b.Method);
  234. }
  235. return b;
  236. }
  237. internal virtual Expression VisitTypeIs(TypeBinaryExpression b) {
  238. Expression expr = this.Visit(b.Expression);
  239. if (expr != b.Expression) {
  240. return Expression.TypeIs(expr, b.TypeOperand);
  241. }
  242. return b;
  243. }
  244. internal virtual Expression VisitConstant(ConstantExpression c) {
  245. return c;
  246. }
  247. internal virtual Expression VisitConditional(ConditionalExpression c) {
  248. Expression test = this.Visit(c.Test);
  249. Expression ifTrue = this.Visit(c.IfTrue);
  250. Expression ifFalse = this.Visit(c.IfFalse);
  251. if (test != c.Test || ifTrue != c.IfTrue || ifFalse != c.IfFalse) {
  252. return Expression.Condition(test, ifTrue, ifFalse);
  253. }
  254. return c;
  255. }
  256. internal virtual Expression VisitParameter(ParameterExpression p) {
  257. return p;
  258. }
  259. internal virtual Expression VisitMemberAccess(MemberExpression m) {
  260. Expression exp = this.Visit(m.Expression);
  261. if (exp != m.Expression) {
  262. return Expression.MakeMemberAccess(exp, m.Member);
  263. }
  264. return m;
  265. }
  266. internal virtual Expression VisitMethodCall(MethodCallExpression m) {
  267. Expression obj = this.Visit(m.Object);
  268. IEnumerable<Expression> args = this.VisitExpressionList(m.Arguments);
  269. if (obj != m.Object || args != m.Arguments) {
  270. return Expression.Call(obj, m.Method, args);
  271. }
  272. return m;
  273. }
  274. internal virtual ReadOnlyCollection<Expression> VisitExpressionList(ReadOnlyCollection<Expression> original) {
  275. List<Expression> list = null;
  276. for (int i = 0, n = original.Count; i < n; i++) {
  277. Expression p = this.Visit(original[i]);
  278. if (list != null) {
  279. list.Add(p);
  280. }
  281. else if (p != original[i]) {
  282. list = new List<Expression>(n);
  283. for (int j = 0; j < i; j++) {
  284. list.Add(original[j]);
  285. }
  286. list.Add(p);
  287. }
  288. }
  289. if (list != null)
  290. return new ReadOnlyCollection<Expression>(list);
  291. return original;
  292. }
  293. internal virtual MemberAssignment VisitMemberAssignment(MemberAssignment assignment) {
  294. Expression e = this.Visit(assignment.Expression);
  295. if (e != assignment.Expression) {
  296. return Expression.Bind(assignment.Member, e);
  297. }
  298. return assignment;
  299. }
  300. internal virtual MemberMemberBinding VisitMemberMemberBinding(MemberMemberBinding binding) {
  301. IEnumerable<MemberBinding> bindings = this.VisitBindingList(binding.Bindings);
  302. if (bindings != binding.Bindings) {
  303. return Expression.MemberBind(binding.Member, bindings);
  304. }
  305. return binding;
  306. }
  307. internal virtual MemberListBinding VisitMemberListBinding(MemberListBinding binding) {
  308. IEnumerable<ElementInit> initializers = this.VisitElementInitializerList(binding.Initializers);
  309. if (initializers != binding.Initializers) {
  310. return Expression.ListBind(binding.Member, initializers);
  311. }
  312. return binding;
  313. }
  314. internal virtual IEnumerable<MemberBinding> VisitBindingList(ReadOnlyCollection<MemberBinding> original) {
  315. List<MemberBinding> list = null;
  316. for (int i = 0, n = original.Count; i < n; i++) {
  317. MemberBinding b = this.VisitBinding(original[i]);
  318. if (list != null) {
  319. list.Add(b);
  320. }
  321. else if (b != original[i]) {
  322. list = new List<MemberBinding>(n);
  323. for (int j = 0; j < i; j++) {
  324. list.Add(original[j]);
  325. }
  326. list.Add(b);
  327. }
  328. }
  329. if (list != null)
  330. return list;
  331. return original;
  332. }
  333. internal virtual IEnumerable<ElementInit> VisitElementInitializerList(ReadOnlyCollection<ElementInit> original) {
  334. List<ElementInit> list = null;
  335. for (int i = 0, n = original.Count; i < n; i++) {
  336. ElementInit init = this.VisitElementInitializer(original[i]);
  337. if (list != null) {
  338. list.Add(init);
  339. }
  340. else if (init != original[i]) {
  341. list = new List<ElementInit>(n);
  342. for (int j = 0; j < i; j++) {
  343. list.Add(original[j]);
  344. }
  345. list.Add(init);
  346. }
  347. }
  348. if (list != null) {
  349. return list;
  350. }
  351. return original;
  352. }
  353. internal virtual Expression VisitLambda(LambdaExpression lambda) {
  354. Expression body = this.Visit(lambda.Body);
  355. if (body != lambda.Body) {
  356. return Expression.Lambda(lambda.Type, body, lambda.Parameters);
  357. }
  358. return lambda;
  359. }
  360. internal virtual NewExpression VisitNew(NewExpression nex) {
  361. IEnumerable<Expression> args = this.VisitExpressionList(nex.Arguments);
  362. if (args != nex.Arguments) {
  363. if (nex.Members != null) {
  364. return Expression.New(nex.Constructor, args, nex.Members);
  365. }
  366. else {
  367. return Expression.New(nex.Constructor, args);
  368. }
  369. }
  370. return nex;
  371. }
  372. internal virtual Expression VisitMemberInit(MemberInitExpression init) {
  373. NewExpression n = this.VisitNew(init.NewExpression);
  374. IEnumerable<MemberBinding> bindings = this.VisitBindingList(init.Bindings);
  375. if (n != init.NewExpression || bindings != init.Bindings) {
  376. return Expression.MemberInit(n, bindings);
  377. }
  378. return init;
  379. }
  380. internal virtual Expression VisitListInit(ListInitExpression init) {
  381. NewExpression n = this.VisitNew(init.NewExpression);
  382. IEnumerable<ElementInit> initializers = this.VisitElementInitializerList(init.Initializers);
  383. if (n != init.NewExpression || initializers != init.Initializers) {
  384. return Expression.ListInit(n, initializers);
  385. }
  386. return init;
  387. }
  388. internal virtual Expression VisitNewArray(NewArrayExpression na) {
  389. IEnumerable<Expression> exprs = this.VisitExpressionList(na.Expressions);
  390. if (exprs != na.Expressions) {
  391. if (na.NodeType == ExpressionType.NewArrayInit) {
  392. return Expression.NewArrayInit(na.Type.GetElementType(), exprs);
  393. }
  394. else {
  395. return Expression.NewArrayBounds(na.Type.GetElementType(), exprs);
  396. }
  397. }
  398. return na;
  399. }
  400. internal virtual Expression VisitInvocation(InvocationExpression iv) {
  401. IEnumerable<Expression> args = this.VisitExpressionList(iv.Arguments);
  402. Expression expr = this.Visit(iv.Expression);
  403. if (args != iv.Arguments || expr != iv.Expression) {
  404. return Expression.Invoke(expr, args);
  405. }
  406. return iv;
  407. }
  408. }
  409. }