using System; using System.Collections.Generic; using System.Linq; using System.Reflection; using System.Security.Cryptography; using System.Text; namespace Cosmos.Assembler { public static class LabelName { /// /// Cache for label names. /// private static Dictionary labelNamesCache = new Dictionary(); // All label naming code should be changed to use this class. // Label bases can be up to 200 chars. If larger they will be shortened with an included hash. // This leaves up to 56 chars for suffix information. // Suffixes are a series of tags and have their own prefixes to preserve backwards compat. // .GUID_xxxxxx // .IL_0000 // .ASM_00 - future, currently is IL_0000 or IL_0000.00 // Would be nice to combine IL and ASM into IL_0000_00, but the way we work with the assembler currently // we cant because the ASM labels are issued as local labels. // // - Methods use a variety of alphanumeric suffixes for support code. // - .00 - asm markers at beginning of method // - .0000.00 IL.ASM marker public static int LabelCount { get; private set; } // Max length of labels at 256. We use lower here so that we still have room for suffixes for IL positions, etc. const int MaxLengthWithoutSuffix = 200; public static string Get(MethodBase aMethod) { string result; if (labelNamesCache.TryGetValue(aMethod, out result)) { return result; } result = Final(GenerateFullName(aMethod)); labelNamesCache.Add(aMethod, result); return result; } public static string Get(string aMethodLabel, int aIlPos) { return aMethodLabel + ".IL_" + aIlPos.ToString("X4"); } // no array bracket, they need to replace, for unique names for used types in methods public static System.Text.RegularExpressions.Regex IllegalCharsReplace = new System.Text.RegularExpressions.Regex(@"[&.,+$<>{}\-\`\\'/\\ \(\)\*!=]", System.Text.RegularExpressions.RegexOptions.Compiled); public static string Final(string xName) { //var xSB = new StringBuilder(xName); // DataMember.FilterStringForIncorrectChars also does some filtering but replacing empties or non _ chars // causes issues with legacy hardcoded values. So we have a separate function. // // For logging possibilities, we generate fuller names, and then strip out spacing/characters. /*const string xIllegalChars = "&.,+$<>{}-`\'/\\ ()[]*!=_"; foreach (char c in xIllegalChars) { xSB.Replace(c.ToString(), ""); }*/ xName = xName.Replace("[]", "array"); xName = xName.Replace("[,]", "array"); xName = xName.Replace("*", "pointer"); xName = IllegalCharsReplace.Replace(xName, string.Empty); if (xName.Length > MaxLengthWithoutSuffix) { using (var xHash = MD5.Create()) { var xValue = xHash.ComputeHash(Encoding.Default.GetBytes(xName)); var xSB = new StringBuilder(xName); // Keep length max same as before. xSB.Length = MaxLengthWithoutSuffix - xValue.Length * 2; foreach (var xByte in xValue) { xSB.Append(xByte.ToString("X2")); } xName = xSB.ToString(); } } LabelCount++; return xName; } public static string GetFullName(Type aType) { if (aType.IsGenericParameter) { return aType.FullName; } StringBuilder xSB = new StringBuilder(256); if (aType.IsArray) { xSB.Append(GetFullName(aType.GetElementType())); xSB.Append("["); int xRank = aType.GetArrayRank(); while (xRank > 1) { xSB.Append(","); xRank--; } xSB.Append("]"); return xSB.ToString(); } if (aType.IsByRef && aType.HasElementType) { return "&" + GetFullName(aType.GetElementType()); } if (aType.IsGenericType && !aType.IsGenericTypeDefinition) { xSB.Append(GetFullName(aType.GetGenericTypeDefinition())); } else { xSB.Append(aType.FullName); } if (aType.IsGenericType) { xSB.Append("<"); var xArgs = aType.GetGenericArguments(); for (int i = 0; i < xArgs.Length - 1; i++) { xSB.Append(GetFullName(xArgs[i])); xSB.Append(", "); } xSB.Append(GetFullName(xArgs.Last())); xSB.Append(">"); } return xSB.ToString(); } public static string GenerateFullName(MethodBase aMethod) { if (aMethod == null) { throw new ArgumentNullException("aMethod"); } var xBuilder = new StringBuilder(256); var xParts = aMethod.ToString().Split(' '); var xParts2 = xParts.Skip(1).ToArray(); var xMethodInfo = aMethod as System.Reflection.MethodInfo; if (xMethodInfo != null) { xBuilder.Append(GetFullName(xMethodInfo.ReturnType)); } else { var xCtor = aMethod as ConstructorInfo; if (xCtor != null) { xBuilder.Append(typeof(void).FullName); } else { xBuilder.Append(xParts[0]); } } xBuilder.Append(" "); if (aMethod.DeclaringType != null) { xBuilder.Append(GetFullName(aMethod.DeclaringType)); } else { xBuilder.Append("dynamic_method"); } xBuilder.Append("."); xBuilder.Append(aMethod.Name); if (aMethod.IsGenericMethod || aMethod.IsGenericMethodDefinition) { var xGenArgs = aMethod.GetGenericArguments(); if (xGenArgs.Length > 0) { xBuilder.Append("<"); for (int i = 0; i < xGenArgs.Length - 1; i++) { xBuilder.Append(GetFullName(xGenArgs[i])); xBuilder.Append(", "); } xBuilder.Append(GetFullName(xGenArgs.Last())); xBuilder.Append(">"); } } xBuilder.Append("("); var xParams = aMethod.GetParameters(); for (var i = 0; i < xParams.Length; i++) { if (i == 0 && xParams[i].Name == "aThis") { continue; } xBuilder.Append(GetFullName(xParams[i].ParameterType)); if (i < (xParams.Length - 1)) { xBuilder.Append(", "); } } xBuilder.Append(")"); return String.Intern(xBuilder.ToString()); } public static string GetFullName(FieldInfo aField) { return GetFullName(aField.FieldType) + " " + GetFullName(aField.DeclaringType) + "." + aField.Name; } } }