Free cookie consent management tool by TermsFeed Policy Generator

source: trunk/sources/HeuristicLab.Persistence/3.3/Default/CompositeSerializers/Storable/StorableReflection.cs @ 3606

Last change on this file since 3606 was 3606, checked in by epitzer, 14 years ago

remove 32bit guard [new code generation works for all configurations (Release/Debug) x (x86/x64/Any)] (#548)

File size: 7.8 KB
Line 
1using System;
2using System.Collections.Generic;
3using System.Linq;
4using HeuristicLab.Persistence.Interfaces;
5using HeuristicLab.Persistence.Core;
6using System.Reflection;
7using HeuristicLab.Persistence.Auxiliary;
8using System.Text;
9using System.Reflection.Emit;
10
11namespace HeuristicLab.Persistence.Default.CompositeSerializers.Storable {
12
13  internal static class StorableReflection {
14
15    private const BindingFlags DECLARED_INSTANCE_MEMBERS =
16      BindingFlags.Instance |
17      BindingFlags.Public |
18      BindingFlags.NonPublic |
19      BindingFlags.DeclaredOnly;
20
21    private delegate void HookWrapper<T>(T o);
22    public delegate void Hook(object o);
23
24    public static IEnumerable<StorableMemberInfo> GenerateStorableMembers(Type type) {
25      var storableMembers = new List<StorableMemberInfo>();
26      if (type.BaseType != null)
27        storableMembers.AddRange(GenerateStorableMembers(type.BaseType));
28
29      var storableClassAttribute = GetStorableClassAttribute(type);
30      if (storableClassAttribute != null) {
31        switch (storableClassAttribute.Type) {
32          case StorableClassType.MarkedOnly:
33            AddMarkedMembers(type, storableMembers); break;
34          case StorableClassType.AllFields:
35            AddAll(type, MemberTypes.Field, storableMembers); break;
36          case StorableClassType.AllProperties:
37            AddAll(type, MemberTypes.Property, storableMembers); break;
38          case StorableClassType.AllFieldsAndAllProperties:
39            AddAll(type, MemberTypes.Field | MemberTypes.Property, storableMembers); break;
40          default:
41            throw new PersistenceException("unsupported [StorableClassType]: " + storableClassAttribute.Type);
42        }
43      }
44
45      return DisentangleNameMapping(storableMembers);
46    }
47
48    public static bool IsEmptyOrStorableType(Type type, bool recursive) {     
49      if (!HasStorableClassAttribute(type) && !IsEmptyType(type, false)) return false;
50      return !recursive || type.BaseType == null || IsEmptyOrStorableType(type.BaseType, true);
51    }
52
53    private static object[] emptyArgs = new object[0];
54
55    public static IEnumerable<Hook> CollectHooks(HookType hookType, Type type) {
56      if (type.BaseType != null)
57        foreach (var mi in CollectHooks(hookType, type.BaseType))
58          yield return mi;
59      foreach (MemberInfo memberInfo in type.GetMembers(DECLARED_INSTANCE_MEMBERS)) {
60        foreach (StorableHookAttribute hook in memberInfo.GetCustomAttributes(typeof(StorableHookAttribute), false)) {
61          if (hook != null && hook.HookType == hookType) {
62            MethodInfo methodInfo = memberInfo as MethodInfo;
63            if (memberInfo.MemberType != MemberTypes.Method || memberInfo == null)
64              throw new ArgumentException("Storable hooks must be methods");           
65            DynamicMethod dm = new DynamicMethod("", null, new[] { typeof(object) }, type);
66            ILGenerator ilgen = dm.GetILGenerator();
67            ilgen.Emit(OpCodes.Ldarg_0);
68            ilgen.Emit(OpCodes.Callvirt, methodInfo);
69            ilgen.Emit(OpCodes.Ret);
70            yield return (Hook)dm.CreateDelegate(typeof(Hook));           
71          }
72        }
73      }
74    }
75
76    #region [Storable] helpers
77
78    private static void AddMarkedMembers(Type type, List<StorableMemberInfo> storableMembers) {
79      foreach (MemberInfo memberInfo in type.GetMembers(DECLARED_INSTANCE_MEMBERS)) {
80        foreach (StorableAttribute attribute in memberInfo.GetCustomAttributes(typeof(StorableAttribute), false)) {
81          storableMembers.Add(new StorableMemberInfo(attribute, memberInfo));
82        }
83      }
84    }
85
86    private static void AddAll(Type type, MemberTypes memberTypes, List<StorableMemberInfo> storableMembers) {
87      foreach (MemberInfo memberInfo in type.GetMembers(DECLARED_INSTANCE_MEMBERS)) {
88        if ((memberInfo.MemberType & memberTypes) == memberInfo.MemberType &&
89            !memberInfo.Name.StartsWith("<") &&
90            !memberInfo.Name.EndsWith("k__BackingField"))
91          storableMembers.Add(new StorableMemberInfo(memberInfo));
92      }
93    }
94
95
96    private static IEnumerable<StorableMemberInfo> DisentangleNameMapping(
97        IEnumerable<StorableMemberInfo> storableMemberInfos) {
98      var nameGrouping = new Dictionary<string, List<StorableMemberInfo>>();
99      foreach (StorableMemberInfo storable in storableMemberInfos) {
100        if (!nameGrouping.ContainsKey(storable.MemberInfo.Name))
101          nameGrouping[storable.MemberInfo.Name] = new List<StorableMemberInfo>();
102        nameGrouping[storable.MemberInfo.Name].Add(storable);
103      }
104      var memberInfos = new List<StorableMemberInfo>();
105      foreach (var storableMemberInfoGroup in nameGrouping.Values) {
106        if (storableMemberInfoGroup.Count == 1) {
107          storableMemberInfoGroup[0].SetDisentangledName(storableMemberInfoGroup[0].MemberInfo.Name);
108          memberInfos.Add(storableMemberInfoGroup[0]);
109        } else if (storableMemberInfoGroup[0].MemberInfo.MemberType == MemberTypes.Field) {
110          foreach (var storableMemberInfo in storableMemberInfoGroup) {
111            storableMemberInfo.SetDisentangledName(storableMemberInfo.FullyQualifiedMemberName);
112            memberInfos.Add(storableMemberInfo);
113          }
114        } else {
115          memberInfos.AddRange(MergePropertyAccessors(storableMemberInfoGroup));
116        }
117      }
118      return memberInfos;
119    }
120
121    private static IEnumerable<StorableMemberInfo> MergePropertyAccessors(List<StorableMemberInfo> members) {
122      var uniqueAccessors = new Dictionary<Type, StorableMemberInfo>();
123      foreach (var member in members)
124        uniqueAccessors[member.GetPropertyDeclaringBaseType()] = member;
125      if (uniqueAccessors.Count == 1) {
126        var storableMemberInfo = uniqueAccessors.Values.First();
127        storableMemberInfo.SetDisentangledName(storableMemberInfo.MemberInfo.Name);
128        yield return storableMemberInfo;
129      } else {
130        foreach (var attribute in uniqueAccessors.Values) {
131          attribute.SetDisentangledName(attribute.FullyQualifiedMemberName);
132          yield return attribute;
133        }
134      }
135    }
136
137    #endregion
138
139    #region [StorableClass] helpers
140
141    private static StorableClassAttribute GetStorableClassAttribute(Type type) {
142      lock (storableClassCache) {
143        if (storableClassCache.ContainsKey(type))
144          return storableClassCache[type];
145        StorableClassAttribute attribute = type
146          .GetCustomAttributes(typeof(StorableClassAttribute), false)
147          .SingleOrDefault() as StorableClassAttribute;
148        storableClassCache.Add(type, attribute);
149        return attribute;
150      }
151    }
152
153    public static bool HasStorableClassAttribute(Type type) {
154      return GetStorableClassAttribute(type) != null;
155    }
156
157    private static Dictionary<Type, StorableClassAttribute> storableClassCache =
158      new Dictionary<Type, StorableClassAttribute>();
159
160    #endregion
161
162    #region other helpers
163
164    private static bool IsEmptyType(Type type, bool recursive) {
165      foreach (MemberInfo memberInfo in type.GetMembers(DECLARED_INSTANCE_MEMBERS)) {
166        if (IsModifiableMember(memberInfo)) return false;
167      }
168      return !recursive || type.BaseType == null || IsEmptyType(type.BaseType, true);
169    }
170
171    private static bool IsModifiableMember(MemberInfo memberInfo) {
172      return memberInfo.MemberType == MemberTypes.Field && IsModifiableField((FieldInfo)memberInfo) ||
173             memberInfo.MemberType == MemberTypes.Property && IsModifiableProperty((PropertyInfo)memberInfo);
174    }
175
176    private static bool IsModifiableField(FieldInfo fi) {
177      return !fi.IsLiteral && !fi.IsInitOnly;
178    }
179
180    private static bool IsModifiableProperty(PropertyInfo pi) {
181      return pi.CanWrite;
182    }
183
184    #endregion
185
186  }
187}
Note: See TracBrowser for help on using the repository browser.