mirror of https://github.com/icsharpcode/ILSpy.git
				
				
			
			You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							534 lines
						
					
					
						
							18 KiB
						
					
					
				
			
		
		
	
	
							534 lines
						
					
					
						
							18 KiB
						
					
					
				// Copyright (c) 2011 AlphaSierraPapa for the SharpDevelop Team | 
						|
//  | 
						|
// Permission is hereby granted, free of charge, to any person obtaining a copy of this | 
						|
// software and associated documentation files (the "Software"), to deal in the Software | 
						|
// without restriction, including without limitation the rights to use, copy, modify, merge, | 
						|
// publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons | 
						|
// to whom the Software is furnished to do so, subject to the following conditions: | 
						|
//  | 
						|
// The above copyright notice and this permission notice shall be included in all copies or | 
						|
// substantial portions of the Software. | 
						|
//  | 
						|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, | 
						|
// INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR | 
						|
// PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE | 
						|
// FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR | 
						|
// OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER | 
						|
// DEALINGS IN THE SOFTWARE. | 
						|
 | 
						|
using System; | 
						|
using System.Collections.Generic; | 
						|
using Mono.Cecil; | 
						|
using Mono.Cecil.Cil; | 
						|
using System.Text; | 
						|
 | 
						|
namespace ICSharpCode.Decompiler.Disassembler | 
						|
{ | 
						|
	public enum ILNameSyntax | 
						|
	{ | 
						|
		/// <summary> | 
						|
		/// class/valuetype + TypeName (built-in types use keyword syntax) | 
						|
		/// </summary> | 
						|
		Signature, | 
						|
		/// <summary> | 
						|
		/// Like signature, but always refers to type parameters using their position | 
						|
		/// </summary> | 
						|
		SignatureNoNamedTypeParameters, | 
						|
		/// <summary> | 
						|
		/// [assembly]Full.Type.Name (even for built-in types) | 
						|
		/// </summary> | 
						|
		TypeName, | 
						|
		/// <summary> | 
						|
		/// Name (but built-in types use keyword syntax) | 
						|
		/// </summary> | 
						|
		ShortTypeName | 
						|
	} | 
						|
 | 
						|
	public static class DisassemblerHelpers | 
						|
	{ | 
						|
		public static string OffsetToString(int offset) | 
						|
		{ | 
						|
			return string.Format("IL_{0:x4}", offset); | 
						|
		} | 
						|
		 | 
						|
		public static string OffsetToString(long offset) | 
						|
		{ | 
						|
			return string.Format("IL_{0:x4}", offset); | 
						|
		} | 
						|
 | 
						|
		public static void WriteOffsetReference(ITextOutput writer, Instruction instruction) | 
						|
		{ | 
						|
			if (instruction == null) | 
						|
				writer.Write("null"); | 
						|
			else | 
						|
				writer.WriteReference(OffsetToString(instruction.Offset), instruction); | 
						|
		} | 
						|
 | 
						|
		public static void WriteTo(this ExceptionHandler exceptionHandler, ITextOutput writer) | 
						|
		{ | 
						|
			writer.Write(".try "); | 
						|
			WriteOffsetReference(writer, exceptionHandler.TryStart); | 
						|
			writer.Write('-'); | 
						|
			WriteOffsetReference(writer, exceptionHandler.TryEnd); | 
						|
			writer.Write(' '); | 
						|
			writer.Write(exceptionHandler.HandlerType.ToString().ToLowerInvariant()); | 
						|
			if (exceptionHandler.FilterStart != null) { | 
						|
				writer.Write(' '); | 
						|
				WriteOffsetReference(writer, exceptionHandler.FilterStart); | 
						|
				writer.Write(" handler "); | 
						|
			} | 
						|
			if (exceptionHandler.CatchType != null) { | 
						|
				writer.Write(' '); | 
						|
				exceptionHandler.CatchType.WriteTo(writer); | 
						|
			} | 
						|
			writer.Write(' '); | 
						|
			WriteOffsetReference(writer, exceptionHandler.HandlerStart); | 
						|
			writer.Write('-'); | 
						|
			WriteOffsetReference(writer, exceptionHandler.HandlerEnd); | 
						|
		} | 
						|
 | 
						|
		public static void WriteTo(this Instruction instruction, ITextOutput writer) | 
						|
		{ | 
						|
			writer.WriteDefinition(OffsetToString(instruction.Offset), instruction); | 
						|
			writer.Write(": "); | 
						|
			writer.WriteReference(instruction.OpCode.Name, instruction.OpCode); | 
						|
			if (instruction.Operand != null) { | 
						|
				writer.Write(' '); | 
						|
				if (instruction.OpCode == OpCodes.Ldtoken) { | 
						|
					if (instruction.Operand is MethodReference) | 
						|
						writer.Write("method "); | 
						|
					else if (instruction.Operand is FieldReference) | 
						|
						writer.Write("field "); | 
						|
				} | 
						|
				WriteOperand(writer, instruction.Operand); | 
						|
			} | 
						|
		} | 
						|
 | 
						|
		static void WriteLabelList(ITextOutput writer, Instruction[] instructions) | 
						|
		{ | 
						|
			writer.Write("("); | 
						|
			for (int i = 0; i < instructions.Length; i++) { | 
						|
				if (i != 0) writer.Write(", "); | 
						|
				WriteOffsetReference(writer, instructions[i]); | 
						|
			} | 
						|
			writer.Write(")"); | 
						|
		} | 
						|
 | 
						|
		static string ToInvariantCultureString(object value) | 
						|
		{ | 
						|
			IConvertible convertible = value as IConvertible; | 
						|
			return (null != convertible) | 
						|
				? convertible.ToString(System.Globalization.CultureInfo.InvariantCulture) | 
						|
				: value.ToString(); | 
						|
		} | 
						|
 | 
						|
		public static void WriteTo(this MethodReference method, ITextOutput writer) | 
						|
		{ | 
						|
			if (method.ExplicitThis) { | 
						|
				writer.Write("instance explicit "); | 
						|
			} else if (method.HasThis) { | 
						|
				writer.Write("instance "); | 
						|
			} | 
						|
			if (method.CallingConvention == MethodCallingConvention.VarArg) { | 
						|
				writer.Write("vararg "); | 
						|
			} | 
						|
			method.ReturnType.WriteTo(writer, ILNameSyntax.SignatureNoNamedTypeParameters); | 
						|
			writer.Write(' '); | 
						|
			if (method.DeclaringType != null) { | 
						|
				method.DeclaringType.WriteTo(writer, ILNameSyntax.TypeName); | 
						|
				writer.Write("::"); | 
						|
			} | 
						|
			MethodDefinition md = method as MethodDefinition; | 
						|
			if (md != null && md.IsCompilerControlled) { | 
						|
				writer.WriteReference(Escape(method.Name + "$PST" + method.MetadataToken.ToInt32().ToString("X8")), method); | 
						|
			} else { | 
						|
				writer.WriteReference(Escape(method.Name), method); | 
						|
			} | 
						|
			GenericInstanceMethod gim = method as GenericInstanceMethod; | 
						|
			if (gim != null) { | 
						|
				writer.Write('<'); | 
						|
				for (int i = 0; i < gim.GenericArguments.Count; i++) { | 
						|
					if (i > 0) | 
						|
						writer.Write(", "); | 
						|
					gim.GenericArguments[i].WriteTo(writer); | 
						|
				} | 
						|
				writer.Write('>'); | 
						|
			} | 
						|
			writer.Write("("); | 
						|
			var parameters = method.Parameters; | 
						|
			for (int i = 0; i < parameters.Count; ++i) { | 
						|
				if (i > 0) | 
						|
					writer.Write(", "); | 
						|
				parameters[i].ParameterType.WriteTo(writer, ILNameSyntax.SignatureNoNamedTypeParameters); | 
						|
			} | 
						|
			writer.Write(")"); | 
						|
		} | 
						|
 | 
						|
		static void WriteTo(this FieldReference field, ITextOutput writer) | 
						|
		{ | 
						|
			field.FieldType.WriteTo(writer, ILNameSyntax.SignatureNoNamedTypeParameters); | 
						|
			writer.Write(' '); | 
						|
			field.DeclaringType.WriteTo(writer, ILNameSyntax.TypeName); | 
						|
			writer.Write("::"); | 
						|
			writer.WriteReference(Escape(field.Name), field); | 
						|
		} | 
						|
 | 
						|
		static bool IsValidIdentifierCharacter(char c) | 
						|
		{ | 
						|
			return c == '_' || c == '$' || c == '@' || c == '?' || c == '`'; | 
						|
		} | 
						|
 | 
						|
		static bool IsValidIdentifier(string identifier) | 
						|
		{ | 
						|
			if (string.IsNullOrEmpty(identifier)) | 
						|
				return false; | 
						|
			if (!(char.IsLetter(identifier[0]) || IsValidIdentifierCharacter(identifier[0]))) { | 
						|
				// As a special case, .ctor and .cctor are valid despite starting with a dot | 
						|
				return identifier == ".ctor" || identifier == ".cctor"; | 
						|
			} | 
						|
			for (int i = 1; i < identifier.Length; i++) { | 
						|
				if (!(char.IsLetterOrDigit(identifier[i]) || IsValidIdentifierCharacter(identifier[i]) || identifier[i] == '.')) | 
						|
					return false; | 
						|
			} | 
						|
			return true; | 
						|
		} | 
						|
 | 
						|
		static readonly HashSet<string> ilKeywords = BuildKeywordList( | 
						|
			"abstract", "algorithm", "alignment", "ansi", "any", "arglist", | 
						|
			"array", "as", "assembly", "assert", "at", "auto", "autochar", "beforefieldinit", | 
						|
			"blob", "blob_object", "bool", "brnull", "brnull.s", "brzero", "brzero.s", "bstr", | 
						|
			"bytearray", "byvalstr", "callmostderived", "carray", "catch", "cdecl", "cf", | 
						|
			"char", "cil", "class", "clsid", "const", "currency", "custom", "date", "decimal", | 
						|
			"default", "demand", "deny", "endmac", "enum", "error", "explicit", "extends", "extern", | 
						|
			"false", "famandassem", "family", "famorassem", "fastcall", "fault", "field", "filetime", | 
						|
			"filter", "final", "finally", "fixed", "float", "float32", "float64", "forwardref", | 
						|
			"fromunmanaged", "handler", "hidebysig", "hresult", "idispatch", "il", "illegal", | 
						|
			"implements", "implicitcom", "implicitres", "import", "in", "inheritcheck", "init", | 
						|
			"initonly", "instance", "int", "int16", "int32", "int64", "int8", "interface", "internalcall", | 
						|
			"iunknown", "lasterr", "lcid", "linkcheck", "literal", "localloc", "lpstr", "lpstruct", "lptstr", | 
						|
			"lpvoid", "lpwstr", "managed", "marshal", "method", "modopt", "modreq", "native", "nested", | 
						|
			"newslot", "noappdomain", "noinlining", "nomachine", "nomangle", "nometadata", "noncasdemand", | 
						|
			"noncasinheritance", "noncaslinkdemand", "noprocess", "not", "not_in_gc_heap", "notremotable", | 
						|
			"notserialized", "null", "nullref", "object", "objectref", "opt", "optil", "out", | 
						|
			"permitonly", "pinned", "pinvokeimpl", "prefix1", "prefix2", "prefix3", "prefix4", "prefix5", "prefix6", | 
						|
			"prefix7", "prefixref", "prejitdeny", "prejitgrant", "preservesig", "private", "privatescope", "protected", | 
						|
			"public", "record", "refany", "reqmin", "reqopt", "reqrefuse", "reqsecobj", "request", "retval", | 
						|
			"rtspecialname", "runtime", "safearray", "sealed", "sequential", "serializable", "special", "specialname", | 
						|
			"static", "stdcall", "storage", "stored_object", "stream", "streamed_object", "string", "struct", | 
						|
			"synchronized", "syschar", "sysstring", "tbstr", "thiscall", "tls", "to", "true", "typedref", | 
						|
			"unicode", "unmanaged", "unmanagedexp", "unsigned", "unused", "userdefined", "value", "valuetype", | 
						|
			"vararg", "variant", "vector", "virtual", "void", "wchar", "winapi", "with", "wrapper", | 
						|
 | 
						|
			// These are not listed as keywords in spec, but ILAsm treats them as such | 
						|
			"property", "type", "flags", "callconv", "strict" | 
						|
		); | 
						|
 | 
						|
		static HashSet<string> BuildKeywordList(params string[] keywords) | 
						|
		{ | 
						|
			HashSet<string> s = new HashSet<string>(keywords); | 
						|
			foreach (var field in typeof(OpCodes).GetFields()) { | 
						|
				s.Add(((OpCode)field.GetValue(null)).Name); | 
						|
			} | 
						|
			return s; | 
						|
		} | 
						|
 | 
						|
		public static string Escape(string identifier) | 
						|
		{ | 
						|
			if (IsValidIdentifier(identifier) && !ilKeywords.Contains(identifier)) { | 
						|
				return identifier; | 
						|
			} else { | 
						|
				// The ECMA specification says that ' inside SQString should be ecaped using an octal escape sequence, | 
						|
				// but we follow Microsoft's ILDasm and use \'. | 
						|
				return "'" + EscapeString(identifier).Replace("'", "\\'") + "'"; | 
						|
			} | 
						|
		} | 
						|
 | 
						|
		public static void WriteTo(this TypeReference type, ITextOutput writer, ILNameSyntax syntax = ILNameSyntax.Signature) | 
						|
		{ | 
						|
			ILNameSyntax syntaxForElementTypes = syntax == ILNameSyntax.SignatureNoNamedTypeParameters ? syntax : ILNameSyntax.Signature; | 
						|
			if (type is PinnedType) { | 
						|
				((PinnedType)type).ElementType.WriteTo(writer, syntaxForElementTypes); | 
						|
				writer.Write(" pinned"); | 
						|
			} else if (type is ArrayType) { | 
						|
				ArrayType at = (ArrayType)type; | 
						|
				at.ElementType.WriteTo(writer, syntaxForElementTypes); | 
						|
				writer.Write('['); | 
						|
				writer.Write(string.Join(", ", at.Dimensions)); | 
						|
				writer.Write(']'); | 
						|
			} else if (type is GenericParameter) { | 
						|
				writer.Write('!'); | 
						|
				if (((GenericParameter)type).Owner.GenericParameterType == GenericParameterType.Method) | 
						|
					writer.Write('!'); | 
						|
				if (string.IsNullOrEmpty(type.Name) || type.Name[0] == '!' || syntax == ILNameSyntax.SignatureNoNamedTypeParameters) | 
						|
					writer.Write(((GenericParameter)type).Position.ToString()); | 
						|
				else | 
						|
					writer.Write(Escape(type.Name)); | 
						|
			} else if (type is ByReferenceType) { | 
						|
				((ByReferenceType)type).ElementType.WriteTo(writer, syntaxForElementTypes); | 
						|
				writer.Write('&'); | 
						|
			} else if (type is PointerType) { | 
						|
				((PointerType)type).ElementType.WriteTo(writer, syntaxForElementTypes); | 
						|
				writer.Write('*'); | 
						|
			} else if (type is GenericInstanceType) { | 
						|
				type.GetElementType().WriteTo(writer, syntaxForElementTypes); | 
						|
				writer.Write('<'); | 
						|
				var arguments = ((GenericInstanceType)type).GenericArguments; | 
						|
				for (int i = 0; i < arguments.Count; i++) { | 
						|
					if (i > 0) | 
						|
						writer.Write(", "); | 
						|
					arguments[i].WriteTo(writer, syntaxForElementTypes); | 
						|
				} | 
						|
				writer.Write('>'); | 
						|
			} else if (type is OptionalModifierType) { | 
						|
				((OptionalModifierType)type).ElementType.WriteTo(writer, syntax); | 
						|
				writer.Write(" modopt("); | 
						|
				((OptionalModifierType)type).ModifierType.WriteTo(writer, ILNameSyntax.TypeName); | 
						|
				writer.Write(") "); | 
						|
			} else if (type is RequiredModifierType) { | 
						|
				((RequiredModifierType)type).ElementType.WriteTo(writer, syntax); | 
						|
				writer.Write(" modreq("); | 
						|
				((RequiredModifierType)type).ModifierType.WriteTo(writer, ILNameSyntax.TypeName); | 
						|
				writer.Write(") "); | 
						|
			} else if (type is SentinelType) { | 
						|
				writer.Write("..., "); | 
						|
				((SentinelType)type).ElementType.WriteTo(writer, syntax); | 
						|
			} else { | 
						|
				string name = PrimitiveTypeName(type.FullName); | 
						|
				if (syntax == ILNameSyntax.ShortTypeName) { | 
						|
					if (name != null) | 
						|
						writer.Write(name); | 
						|
					else | 
						|
						writer.WriteReference(Escape(type.Name), type); | 
						|
				} else if ((syntax == ILNameSyntax.Signature || syntax == ILNameSyntax.SignatureNoNamedTypeParameters) && name != null) { | 
						|
					writer.Write(name); | 
						|
				} else { | 
						|
					if (syntax == ILNameSyntax.Signature || syntax == ILNameSyntax.SignatureNoNamedTypeParameters) | 
						|
						writer.Write(type.IsValueType ? "valuetype " : "class "); | 
						|
 | 
						|
					if (type.DeclaringType != null) { | 
						|
						type.DeclaringType.WriteTo(writer, ILNameSyntax.TypeName); | 
						|
						writer.Write('/'); | 
						|
						writer.WriteReference(Escape(type.Name), type); | 
						|
					} else { | 
						|
						if (!type.IsDefinition && type.Scope != null && !(type is TypeSpecification)) | 
						|
							writer.Write("[{0}]", Escape(type.Scope.Name)); | 
						|
						writer.WriteReference(Escape(type.FullName), type); | 
						|
					} | 
						|
				} | 
						|
			} | 
						|
		} | 
						|
 | 
						|
		public static void WriteOperand(ITextOutput writer, object operand) | 
						|
		{ | 
						|
			if (operand == null) | 
						|
				throw new ArgumentNullException(nameof(operand)); | 
						|
 | 
						|
			Instruction targetInstruction = operand as Instruction; | 
						|
			if (targetInstruction != null) { | 
						|
				WriteOffsetReference(writer, targetInstruction); | 
						|
				return; | 
						|
			} | 
						|
 | 
						|
			Instruction[] targetInstructions = operand as Instruction[]; | 
						|
			if (targetInstructions != null) { | 
						|
				WriteLabelList(writer, targetInstructions); | 
						|
				return; | 
						|
			} | 
						|
 | 
						|
			VariableReference variableRef = operand as VariableReference; | 
						|
			if (variableRef != null) { | 
						|
				if (string.IsNullOrEmpty(variableRef.Name)) | 
						|
					writer.WriteReference(variableRef.Index.ToString(), variableRef); | 
						|
				else | 
						|
					writer.WriteReference(Escape(variableRef.Name), variableRef); | 
						|
				return; | 
						|
			} | 
						|
 | 
						|
			ParameterReference paramRef = operand as ParameterReference; | 
						|
			if (paramRef != null) { | 
						|
				if (string.IsNullOrEmpty(paramRef.Name)) | 
						|
					writer.WriteReference(paramRef.Index.ToString(), paramRef); | 
						|
				else | 
						|
					writer.WriteReference(Escape(paramRef.Name), paramRef); | 
						|
				return; | 
						|
			} | 
						|
 | 
						|
			MethodReference methodRef = operand as MethodReference; | 
						|
			if (methodRef != null) { | 
						|
				methodRef.WriteTo(writer); | 
						|
				return; | 
						|
			} | 
						|
 | 
						|
			TypeReference typeRef = operand as TypeReference; | 
						|
			if (typeRef != null) { | 
						|
				typeRef.WriteTo(writer, ILNameSyntax.TypeName); | 
						|
				return; | 
						|
			} | 
						|
 | 
						|
			FieldReference fieldRef = operand as FieldReference; | 
						|
			if (fieldRef != null) { | 
						|
				fieldRef.WriteTo(writer); | 
						|
				return; | 
						|
			} | 
						|
 | 
						|
			string s = operand as string; | 
						|
			if (s != null) { | 
						|
				WriteOperand(writer, s); | 
						|
			} else if (operand is char) { | 
						|
				writer.Write(((int)(char)operand).ToString()); | 
						|
			} else if (operand is float) { | 
						|
				WriteOperand(writer, (float)operand); | 
						|
			} else if (operand is double) { | 
						|
				WriteOperand(writer, (double)operand); | 
						|
			} else if (operand is bool) { | 
						|
				writer.Write((bool)operand ? "true" : "false"); | 
						|
			} else { | 
						|
				s = ToInvariantCultureString(operand); | 
						|
				writer.Write(s); | 
						|
			} | 
						|
		} | 
						|
 | 
						|
		public static void WriteOperand(ITextOutput writer, long val) | 
						|
		{ | 
						|
			writer.Write(ToInvariantCultureString(val)); | 
						|
		} | 
						|
 | 
						|
		public static void WriteOperand(ITextOutput writer, float val) | 
						|
		{ | 
						|
			if (val == 0) { | 
						|
				if (1 / val == float.NegativeInfinity) { | 
						|
					// negative zero is a special case | 
						|
					writer.Write('-'); | 
						|
				} | 
						|
				writer.Write("0.0"); | 
						|
			} else if (float.IsInfinity(val) || float.IsNaN(val)) { | 
						|
				byte[] data = BitConverter.GetBytes(val); | 
						|
				writer.Write('('); | 
						|
				for (int i = 0; i < data.Length; i++) { | 
						|
					if (i > 0) | 
						|
						writer.Write(' '); | 
						|
					writer.Write(data[i].ToString("X2")); | 
						|
				} | 
						|
				writer.Write(')'); | 
						|
			} else { | 
						|
				writer.Write(val.ToString("R", System.Globalization.CultureInfo.InvariantCulture)); | 
						|
			} | 
						|
		} | 
						|
 | 
						|
		public static void WriteOperand(ITextOutput writer, double val) | 
						|
		{ | 
						|
			if (val == 0) { | 
						|
				if (1 / val == double.NegativeInfinity) { | 
						|
					// negative zero is a special case | 
						|
					writer.Write('-'); | 
						|
				} | 
						|
				writer.Write("0.0"); | 
						|
			} else if (double.IsInfinity(val) || double.IsNaN(val)) { | 
						|
				byte[] data = BitConverter.GetBytes(val); | 
						|
				writer.Write('('); | 
						|
				for (int i = 0; i < data.Length; i++) { | 
						|
					if (i > 0) | 
						|
						writer.Write(' '); | 
						|
					writer.Write(data[i].ToString("X2")); | 
						|
				} | 
						|
				writer.Write(')'); | 
						|
			} else { | 
						|
				writer.Write(val.ToString("R", System.Globalization.CultureInfo.InvariantCulture)); | 
						|
			} | 
						|
		} | 
						|
 | 
						|
		public static void WriteOperand(ITextOutput writer, string operand) | 
						|
		{ | 
						|
			writer.Write('"'); | 
						|
			writer.Write(EscapeString(operand)); | 
						|
			writer.Write('"'); | 
						|
		} | 
						|
 | 
						|
		public static string EscapeString(string str) | 
						|
		{ | 
						|
			StringBuilder sb = new StringBuilder(); | 
						|
			foreach (char ch in str) { | 
						|
				switch (ch) { | 
						|
					case '"': | 
						|
						sb.Append("\\\""); | 
						|
						break; | 
						|
					case '\\': | 
						|
						sb.Append("\\\\"); | 
						|
						break; | 
						|
					case '\0': | 
						|
						sb.Append("\\0"); | 
						|
						break; | 
						|
					case '\a': | 
						|
						sb.Append("\\a"); | 
						|
						break; | 
						|
					case '\b': | 
						|
						sb.Append("\\b"); | 
						|
						break; | 
						|
					case '\f': | 
						|
						sb.Append("\\f"); | 
						|
						break; | 
						|
					case '\n': | 
						|
						sb.Append("\\n"); | 
						|
						break; | 
						|
					case '\r': | 
						|
						sb.Append("\\r"); | 
						|
						break; | 
						|
					case '\t': | 
						|
						sb.Append("\\t"); | 
						|
						break; | 
						|
					case '\v': | 
						|
						sb.Append("\\v"); | 
						|
						break; | 
						|
					default: | 
						|
						// print control characters and uncommon white spaces as numbers | 
						|
						if (char.IsControl(ch) || char.IsSurrogate(ch) || (char.IsWhiteSpace(ch) && ch != ' ')) { | 
						|
							sb.Append("\\u" + ((int)ch).ToString("x4")); | 
						|
						} else { | 
						|
							sb.Append(ch); | 
						|
						} | 
						|
						break; | 
						|
				} | 
						|
			} | 
						|
			return sb.ToString(); | 
						|
		} | 
						|
		public static string PrimitiveTypeName(string fullName) | 
						|
		{ | 
						|
			switch (fullName) { | 
						|
				case "System.SByte": | 
						|
					return "int8"; | 
						|
				case "System.Int16": | 
						|
					return "int16"; | 
						|
				case "System.Int32": | 
						|
					return "int32"; | 
						|
				case "System.Int64": | 
						|
					return "int64"; | 
						|
				case "System.Byte": | 
						|
					return "uint8"; | 
						|
				case "System.UInt16": | 
						|
					return "uint16"; | 
						|
				case "System.UInt32": | 
						|
					return "uint32"; | 
						|
				case "System.UInt64": | 
						|
					return "uint64"; | 
						|
				case "System.Single": | 
						|
					return "float32"; | 
						|
				case "System.Double": | 
						|
					return "float64"; | 
						|
				case "System.Void": | 
						|
					return "void"; | 
						|
				case "System.Boolean": | 
						|
					return "bool"; | 
						|
				case "System.String": | 
						|
					return "string"; | 
						|
				case "System.Char": | 
						|
					return "char"; | 
						|
				case "System.Object": | 
						|
					return "object"; | 
						|
				case "System.IntPtr": | 
						|
					return "native int"; | 
						|
				default: | 
						|
					return null; | 
						|
			} | 
						|
		} | 
						|
	} | 
						|
}
 | 
						|
 |