File: CachedExpressionCompiler.cs

package info (click to toggle)
mono 2.6.7-5.1
  • links: PTS, VCS
  • area: main
  • in suites: squeeze
  • size: 327,344 kB
  • ctags: 413,649
  • sloc: cs: 2,471,883; xml: 1,768,594; ansic: 350,665; sh: 13,644; makefile: 8,640; perl: 1,784; asm: 717; cpp: 209; python: 146; sql: 81; sed: 16
file content (86 lines) | stat: -rw-r--r-- 4,049 bytes parent folder | download | duplicates (2)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
/* ****************************************************************************
 *
 * Copyright (c) Microsoft Corporation. All rights reserved.
 *
 * This software is subject to the Microsoft Public License (Ms-PL). 
 * A copy of the license can be found in the license.htm file included 
 * in this distribution.
 *
 * You must not remove this notice, or any other, from this software.
 *
 * ***************************************************************************/

namespace System.Web.Mvc.ExpressionUtil {
    using System;
    using System.Linq.Expressions;

    internal static class CachedExpressionCompiler {

        // This is the entry point to the cached expression tree compiler. The processor will perform a series of checks
        // and optimizations in order to return a fully-compiled func as quickly as possible to the caller. If the
        // input expression is particularly obscure, the system will fall back to a slow but correct compilation step.
        public static Func<TModel, TValue> Process<TModel, TValue>(Expression<Func<TModel, TValue>> lambdaExpression) {
            return Processor<TModel, TValue>.GetFunc(lambdaExpression);
        }

        private static class Processor<TModel, TValue> {

            private static readonly Cache _cache = new Cache();

            public static Func<TModel, TValue> GetFunc(Expression<Func<TModel, TValue>> lambdaExpression) {
                // look for common patterns that don't need to be fingerprinted
                Func<TModel, TValue> func = GetFuncFastTrack(lambdaExpression);
                if (func != null) {
                    return func;
                }

                // not a common pattern, so try fingerprinting (slower, but cached)
                func = GetFuncFingerprinted(lambdaExpression);
                if (func != null) {
                    return func;
                }

                // pattern not recognized by fingerprinting routine, so compile directly (slowest)
                return GetFuncSlow(lambdaExpression);
            }

            private static Func<TModel, TValue> GetFuncFastTrack(Expression<Func<TModel, TValue>> lambdaExpression) {
                ParameterExpression modelParameter = lambdaExpression.Parameters[0];
                Expression body = lambdaExpression.Body;

                return FastTrack<TModel, TValue>.GetFunc(modelParameter, body);
            }

            private static Func<TModel, TValue> GetFuncFingerprinted(Expression<Func<TModel, TValue>> lambdaExpression) {
                ParserContext context = ExpressionParser.Parse(lambdaExpression);
                if (context.Fingerprint == null) {
                    // fingerprinting failed
                    return null;
                }

                object[] hoistedValues = context.HoistedValues.ToArray();
                var del = _cache.GetDelegate(context);
                return model => del(model, hoistedValues);
            }

            private static Func<TModel, TValue> GetFuncSlow(Expression<Func<TModel, TValue>> lambdaExpression) {
                Func<TModel, TValue> del = lambdaExpression.Compile();
                return del;
            }

            private sealed class Cache : ReaderWriterCache<ExpressionFingerprint, CompiledExpressionDelegate<TModel, TValue>> {
                private static CompiledExpressionDelegate<TModel, TValue> CreateDelegate(ParserContext context) {
                    var bodyExpr = context.Fingerprint.ToExpression(context);
                    var lambdaExpr = Expression.Lambda<CompiledExpressionDelegate<TModel, TValue>>(bodyExpr, context.ModelParameter, ParserContext.HoistedValuesParameter);
                    var del = lambdaExpr.Compile();
                    return del;
                }
                public CompiledExpressionDelegate<TModel, TValue> GetDelegate(ParserContext context) {
                    return FetchOrCreateItem(context.Fingerprint, () => CreateDelegate(context));
                }
            }

        }

    }
}