/* Copyright (C) 2007-2014 Jeroen Frijters Copyright (C) 2009 Volker Berlin (i-net software) This software is provided 'as-is', without any express or implied warranty. In no event will the authors be held liable for any damages arising from the use of this software. Permission is granted to anyone to use this software for any purpose, including commercial applications, and to alter it and redistribute it freely, subject to the following restrictions: 1. The origin of this software must not be misrepresented; you must not claim that you wrote the original software. If you use this software in a product, an acknowledgment in the product documentation would be appreciated but is not required. 2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software. 3. This notice may not be removed or altered from any source distribution. Jeroen Frijters jeroen@frijters.net */ using System; using System.Collections.Generic; using System.Diagnostics; using System.Reflection; using System.Reflection.Emit; using System.Runtime.ConstrainedExecution; using System.Runtime.InteropServices; using System.Runtime.Serialization; using System.Security; using System.Threading; using IKVM.Internal; static class Java_sun_misc_GC { public static long maxObjectInspectionAge() { return 0; } } static class Java_sun_misc_MessageUtils { public static void toStderr(string msg) { Console.Error.Write(msg); } public static void toStdout(string msg) { Console.Out.Write(msg); } } static class Java_sun_misc_MiscHelper { public static object getAssemblyClassLoader(Assembly asm, object extcl) { if (extcl == null || asm.IsDefined(typeof(IKVM.Attributes.CustomAssemblyClassLoaderAttribute), false)) { return AssemblyClassLoader.FromAssembly(asm).GetJavaClassLoader(); } return null; } } static class Java_sun_misc_Signal { /* derived from version 6.0 VC98/include/signal.h */ private const int SIGINT = 2; /* interrupt */ private const int SIGILL = 4; /* illegal instruction - invalid function image */ private const int SIGFPE = 8; /* floating point exception */ private const int SIGSEGV = 11; /* segment violation */ private const int SIGTERM = 15; /* Software termination signal from kill */ private const int SIGBREAK = 21; /* Ctrl-Break sequence */ private const int SIGABRT = 22; /* abnormal termination triggered by abort call */ private static Dictionary handler = new Dictionary(); // Delegate type to be used as the Handler Routine for SetConsoleCtrlHandler private delegate Boolean ConsoleCtrlDelegate(CtrlTypes CtrlType); // Enumerated type for the control messages sent to the handler routine private enum CtrlTypes : uint { CTRL_C_EVENT = 0, CTRL_BREAK_EVENT, CTRL_CLOSE_EVENT, CTRL_LOGOFF_EVENT = 5, CTRL_SHUTDOWN_EVENT } [SecurityCritical] private sealed class CriticalCtrlHandler : CriticalFinalizerObject { private ConsoleCtrlDelegate consoleCtrlDelegate; private bool ok; [DllImport("kernel32.dll")] private static extern bool SetConsoleCtrlHandler(ConsoleCtrlDelegate e, bool add); internal CriticalCtrlHandler() { consoleCtrlDelegate = new ConsoleCtrlDelegate(ConsoleCtrlCheck); ok = SetConsoleCtrlHandler(consoleCtrlDelegate, true); } [SecuritySafeCritical] ~CriticalCtrlHandler() { if (ok) { SetConsoleCtrlHandler(consoleCtrlDelegate, false); } } } private static object defaultConsoleCtrlDelegate; private static bool ConsoleCtrlCheck(CtrlTypes ctrlType) { #if !FIRST_PASS switch (ctrlType) { case CtrlTypes.CTRL_BREAK_EVENT: DumpAllJavaThreads(); return true; } #endif return false; } #if !FIRST_PASS private static void DumpAllJavaThreads() { Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss")); java.util.Map traces = java.lang.Thread.getAllStackTraces(); Console.WriteLine("Full thread dump IKVM.NET {0} ({1} bit):", JVM.SafeGetAssemblyVersion(Assembly.GetExecutingAssembly()), IntPtr.Size * 8); java.util.Iterator entries = traces.entrySet().iterator(); while (entries.hasNext()) { java.util.Map.Entry entry = (java.util.Map.Entry)entries.next(); java.lang.Thread thread = (java.lang.Thread)entry.getKey(); Console.WriteLine("\n\"{0}\"{1} prio={2} tid=0x{3:X8}", thread.getName(), thread.isDaemon() ? " daemon" : "", thread.getPriority(), thread.getId()); Console.WriteLine(" java.lang.Thread.State: " + thread.getState()); java.lang.StackTraceElement[] trace = (java.lang.StackTraceElement[])entry.getValue(); for (int i = 0; i < trace.Length; i++) { Console.WriteLine("\tat {0}", trace[i]); } } Console.WriteLine(); } #endif public static int findSignal(string sigName) { if (Environment.OSVersion.Platform == PlatformID.Win32NT) { switch (sigName) { case "ABRT": /* abnormal termination triggered by abort cl */ return SIGABRT; case "FPE": /* floating point exception */ return SIGFPE; case "SEGV": /* segment violation */ return SIGSEGV; case "INT": /* interrupt */ return SIGINT; case "TERM": /* software term signal from kill */ return SIGTERM; case "BREAK": /* Ctrl-Break sequence */ return SIGBREAK; case "ILL": /* illegal instruction */ return SIGILL; } } return -1; } // this is a separate method to be able to catch the SecurityException (for the LinkDemand) [SecuritySafeCritical] private static void RegisterCriticalCtrlHandler() { defaultConsoleCtrlDelegate = new CriticalCtrlHandler(); } // Register a signal handler public static long handle0(int sig, long nativeH) { long oldHandler; handler.TryGetValue(sig, out oldHandler); switch (nativeH) { case 0: // Default Signal Handler if (defaultConsoleCtrlDelegate == null && Environment.OSVersion.Platform == PlatformID.Win32NT) { try { RegisterCriticalCtrlHandler(); } catch (SecurityException) { } } break; case 1: // Ignore Signal break; case 2: // Custom Signal Handler switch (sig) { case SIGBREAK: case SIGFPE: return -1; } break; } handler[sig] = nativeH; return oldHandler; } public static void raise0(int sig) { #if !FIRST_PASS java.security.AccessController.doPrivileged(ikvm.runtime.Delegates.toPrivilegedAction(delegate { java.lang.Class clazz = typeof(sun.misc.Signal); java.lang.reflect.Method dispatch = clazz.getDeclaredMethod("dispatch", java.lang.Integer.TYPE); dispatch.setAccessible(true); dispatch.invoke(null, java.lang.Integer.valueOf(sig)); return null; })); #endif } } static class Java_sun_misc_NativeSignalHandler { public static void handle0(int number, long handler) { throw new NotImplementedException(); } } static class Java_sun_misc_Perf { public static object attach(object thisPerf, string user, int lvmid, int mode) { throw new NotImplementedException(); } public static void detach(object thisPerf, object bb) { throw new NotImplementedException(); } public static object createLong(object thisPerf, string name, int variability, int units, long value) { #if FIRST_PASS return null; #else return java.nio.ByteBuffer.allocate(8); #endif } public static object createByteArray(object thisPerf, string name, int variability, int units, byte[] value, int maxLength) { #if FIRST_PASS return null; #else return java.nio.ByteBuffer.allocate(maxLength).put(value); #endif } public static long highResCounter(object thisPerf) { throw new NotImplementedException(); } public static long highResFrequency(object thisPerf) { throw new NotImplementedException(); } public static void registerNatives() { } } static class Java_sun_misc_Unsafe { private static void CheckArrayBounds(object obj, long offset, int accessLength) { // NOTE we rely on the fact that Buffer.ByteLength() requires a primitive array int arrayLength = Buffer.ByteLength((Array)obj); if (offset < 0 || offset > arrayLength - accessLength || accessLength > arrayLength) { throw new IndexOutOfRangeException(); } } [SecuritySafeCritical] public static short ReadInt16(object obj, long offset) { Stats.Log("ReadInt16"); CheckArrayBounds(obj, offset, 2); GCHandle handle = GCHandle.Alloc(obj, GCHandleType.Pinned); short value = Marshal.ReadInt16((IntPtr)(handle.AddrOfPinnedObject().ToInt64() + offset)); handle.Free(); return value; } [SecuritySafeCritical] public static int ReadInt32(object obj, long offset) { Stats.Log("ReadInt32"); CheckArrayBounds(obj, offset, 4); GCHandle handle = GCHandle.Alloc(obj, GCHandleType.Pinned); int value = Marshal.ReadInt32((IntPtr)(handle.AddrOfPinnedObject().ToInt64() + offset)); handle.Free(); return value; } [SecuritySafeCritical] public static long ReadInt64(object obj, long offset) { Stats.Log("ReadInt64"); CheckArrayBounds(obj, offset, 8); GCHandle handle = GCHandle.Alloc(obj, GCHandleType.Pinned); long value = Marshal.ReadInt64((IntPtr)(handle.AddrOfPinnedObject().ToInt64() + offset)); handle.Free(); return value; } [SecuritySafeCritical] public static void WriteInt16(object obj, long offset, short value) { Stats.Log("WriteInt16"); CheckArrayBounds(obj, offset, 2); GCHandle handle = GCHandle.Alloc(obj, GCHandleType.Pinned); Marshal.WriteInt16((IntPtr)(handle.AddrOfPinnedObject().ToInt64() + offset), value); handle.Free(); } [SecuritySafeCritical] public static void WriteInt32(object obj, long offset, int value) { Stats.Log("WriteInt32"); CheckArrayBounds(obj, offset, 4); GCHandle handle = GCHandle.Alloc(obj, GCHandleType.Pinned); Marshal.WriteInt32((IntPtr)(handle.AddrOfPinnedObject().ToInt64() + offset), value); handle.Free(); } [SecuritySafeCritical] public static void WriteInt64(object obj, long offset, long value) { Stats.Log("WriteInt64"); CheckArrayBounds(obj, offset, 8); GCHandle handle = GCHandle.Alloc(obj, GCHandleType.Pinned); Marshal.WriteInt64((IntPtr)(handle.AddrOfPinnedObject().ToInt64() + offset), value); handle.Free(); } public static void throwException(object thisUnsafe, Exception x) { throw x; } public static bool shouldBeInitialized(object thisUnsafe, java.lang.Class clazz) { return TypeWrapper.FromClass(clazz).HasStaticInitializer; } public static void ensureClassInitialized(object thisUnsafe, java.lang.Class clazz) { TypeWrapper tw = TypeWrapper.FromClass(clazz); if (!tw.IsArray) { try { tw.Finish(); } catch (RetargetableJavaException x) { throw x.ToJava(); } tw.RunClassInit(); } } [SecurityCritical] public static object allocateInstance(object thisUnsafe, java.lang.Class clazz) { TypeWrapper wrapper = TypeWrapper.FromClass(clazz); try { wrapper.Finish(); } catch (RetargetableJavaException x) { throw x.ToJava(); } return FormatterServices.GetUninitializedObject(wrapper.TypeAsBaseType); } public static java.lang.Class defineClass(object thisUnsafe, string name, byte[] buf, int offset, int length, java.lang.ClassLoader cl, java.security.ProtectionDomain pd) { return Java_java_lang_ClassLoader.defineClass1(cl, name.Replace('/', '.'), buf, offset, length, pd, null); } public static java.lang.Class defineClass(object thisUnsafe, string name, byte[] buf, int offset, int length, ikvm.@internal.CallerID callerID) { #if FIRST_PASS return null; #else return defineClass(thisUnsafe, name, buf, offset, length, callerID.getCallerClassLoader(), callerID.getCallerClass().pd); #endif } public static java.lang.Class defineAnonymousClass(object thisUnsafe, java.lang.Class host, byte[] data, object[] cpPatches) { #if FIRST_PASS return null; #else try { ClassLoaderWrapper loader = TypeWrapper.FromClass(host).GetClassLoader(); ClassFile classFile = new ClassFile(data, 0, data.Length, "", loader.ClassFileParseOptions, cpPatches); if (classFile.IKVMAssemblyAttribute != null) { // if this happens, the OpenJDK is probably trying to load an OpenJDK class file as a resource, // make sure the build process includes the original class file as a resource in that case throw new java.lang.ClassFormatError("Trying to define anonymous class based on stub class: " + classFile.Name); } return loader.GetTypeWrapperFactory().DefineClassImpl(null, TypeWrapper.FromClass(host), classFile, loader, host.pd).ClassObject; } catch (RetargetableJavaException x) { throw x.ToJava(); } #endif } public static bool compareAndSwapInt(object thisUnsafe, object obj, long offset, int expect, int update) { #if FIRST_PASS return false; #else int[] array = obj as int[]; if (array != null && (offset & 3) == 0) { Stats.Log("compareAndSwapInt.array"); return Interlocked.CompareExchange(ref array[offset / 4], update, expect) == expect; } else if (obj is Array) { Stats.Log("compareAndSwapInt.unaligned"); // unaligned or not the right array type, so we can't be atomic lock (thisUnsafe) { if (ReadInt32(obj, offset) == expect) { WriteInt32(obj, offset, update); return true; } return false; } } else { if (offset >= cacheCompareExchangeInt32.Length || cacheCompareExchangeInt32[offset] == null) { InterlockedResize(ref cacheCompareExchangeInt32, (int)offset + 1); cacheCompareExchangeInt32[offset] = (CompareExchangeInt32)CreateCompareExchange(offset); } Stats.Log("compareAndSwapInt.", offset); return cacheCompareExchangeInt32[offset](obj, update, expect) == expect; } #endif } public static bool compareAndSwapLong(object thisUnsafe, object obj, long offset, long expect, long update) { #if FIRST_PASS return false; #else long[] array = obj as long[]; if (array != null && (offset & 7) == 0) { Stats.Log("compareAndSwapLong.array"); return Interlocked.CompareExchange(ref array[offset / 8], update, expect) == expect; } else if (obj is Array) { Stats.Log("compareAndSwapLong.unaligned"); // unaligned or not the right array type, so we can't be atomic lock (thisUnsafe) { if (ReadInt64(obj, offset) == expect) { WriteInt64(obj, offset, update); return true; } return false; } } else { if (offset >= cacheCompareExchangeInt64.Length || cacheCompareExchangeInt64[offset] == null) { InterlockedResize(ref cacheCompareExchangeInt64, (int)offset + 1); cacheCompareExchangeInt64[offset] = (CompareExchangeInt64)CreateCompareExchange(offset); } Stats.Log("compareAndSwapLong.", offset); return cacheCompareExchangeInt64[offset](obj, update, expect) == expect; } #endif } private delegate int CompareExchangeInt32(object obj, int value, int comparand); private delegate long CompareExchangeInt64(object obj, long value, long comparand); private static CompareExchangeInt32[] cacheCompareExchangeInt32 = new CompareExchangeInt32[0]; private static CompareExchangeInt64[] cacheCompareExchangeInt64 = new CompareExchangeInt64[0]; private static void InterlockedResize(ref T[] array, int newSize) { for (; ; ) { T[] oldArray = array; T[] newArray = oldArray; if (oldArray.Length >= newSize) { return; } Array.Resize(ref newArray, newSize); if (Interlocked.CompareExchange(ref array, newArray, oldArray) == oldArray) { return; } } } #if !FIRST_PASS private static Delegate CreateCompareExchange(long fieldOffset) { FieldInfo field = GetFieldInfo(fieldOffset); DynamicMethod dm = new DynamicMethod("CompareExchange", field.FieldType, new Type[] { typeof(object), field.FieldType, field.FieldType }, field.DeclaringType); ILGenerator ilgen = dm.GetILGenerator(); ilgen.Emit(OpCodes.Ldarg_0); ilgen.Emit(OpCodes.Castclass, field.DeclaringType); ilgen.Emit(OpCodes.Ldflda, field); ilgen.Emit(OpCodes.Ldarg_1); ilgen.Emit(OpCodes.Ldarg_2); ilgen.Emit(OpCodes.Call, typeof(Interlocked).GetMethod("CompareExchange", new Type[] { field.FieldType.MakeByRefType(), field.FieldType, field.FieldType })); ilgen.Emit(OpCodes.Ret); return dm.CreateDelegate(field.FieldType == typeof(int) ? typeof(CompareExchangeInt32) : typeof(CompareExchangeInt64)); } private static FieldInfo GetFieldInfo(long offset) { FieldWrapper fw = FieldWrapper.FromField(sun.misc.Unsafe.getField(offset)); fw.Link(); fw.ResolveField(); return fw.GetField(); } #endif public static bool compareAndSwapObject(object thisUnsafe, object obj, long offset, object expect, object update) { #if FIRST_PASS return false; #else object[] array = obj as object[]; if (array != null) { Stats.Log("compareAndSwapObject.array"); return Atomic.CompareExchange(array, (int)offset, update, expect) == expect; } else { Stats.Log("compareAndSwapObject.", offset); return Atomic.CompareExchange(obj, new FieldInfo[] { GetFieldInfo(offset) }, update, expect) == expect; } #endif } abstract class Atomic { // NOTE we don't care that we keep the Type alive, because Unsafe should only be used inside the core class libraries private static Dictionary impls = new Dictionary(); internal static object CompareExchange(object obj, FieldInfo[] field, object value, object comparand) { TypedReference tr = TypedReference.MakeTypedReference(obj, field); return GetImpl(__reftype(tr)).CompareExchangeImpl(tr, value, comparand); } internal static object CompareExchange(object[] array, int index, object value, object comparand) { return GetImpl(array.GetType().GetElementType()).CompareExchangeImpl(array, index, value, comparand); } private static Atomic GetImpl(Type type) { Atomic impl; if (!impls.TryGetValue(type, out impl)) { impl = (Atomic)Activator.CreateInstance(typeof(Impl<>).MakeGenericType(type)); Dictionary curr = impls; Dictionary copy = new Dictionary(curr); copy[type] = impl; Interlocked.CompareExchange(ref impls, copy, curr); } return impl; } protected abstract object CompareExchangeImpl(TypedReference tr, object value, object comparand); protected abstract object CompareExchangeImpl(object[] array, int index, object value, object comparand); sealed class Impl : Atomic where T : class { protected override object CompareExchangeImpl(TypedReference tr, object value, object comparand) { return Interlocked.CompareExchange(ref __refvalue(tr, T), (T)value, (T)comparand); } protected override object CompareExchangeImpl(object[] array, int index, object value, object comparand) { return Interlocked.CompareExchange(ref ((T[])array)[index], (T)value, (T)comparand); } } } static class Stats { #if !FIRST_PASS && UNSAFE_STATISTICS private static readonly Dictionary dict = new Dictionary(); static Stats() { java.lang.Runtime.getRuntime().addShutdownHook(new DumpStats()); } sealed class DumpStats : java.lang.Thread { public override void run() { List> list = new List>(dict); list.Sort(delegate(KeyValuePair kv1, KeyValuePair kv2) { return kv1.Value.CompareTo(kv2.Value); }); foreach (KeyValuePair kv in list) { Console.WriteLine("{0,10}: {1}", kv.Value, kv.Key); } } } #endif [Conditional("UNSAFE_STATISTICS")] internal static void Log(string key) { #if !FIRST_PASS && UNSAFE_STATISTICS lock (dict) { int count; dict.TryGetValue(key, out count); dict[key] = count + 1; } #endif } [Conditional("UNSAFE_STATISTICS")] internal static void Log(string key, long offset) { #if !FIRST_PASS && UNSAFE_STATISTICS FieldWrapper field = FieldWrapper.FromField(sun.misc.Unsafe.getField(offset)); key += field.DeclaringType.Name + "::" + field.Name; Log(key); #endif } } } static class Java_sun_misc_Version { public static string getJvmSpecialVersion() { throw new NotImplementedException(); } public static string getJdkSpecialVersion() { throw new NotImplementedException(); } public static bool getJvmVersionInfo() { throw new NotImplementedException(); } public static void getJdkVersionInfo() { throw new NotImplementedException(); } } static class Java_sun_misc_VM { public static void initialize() { } public static java.lang.ClassLoader latestUserDefinedLoader() { // testing shows that it is cheaper the get the full stack trace and then look at a few frames than getting the frames individually StackTrace trace = new StackTrace(2, false); for (int i = 0; i < trace.FrameCount; i++) { StackFrame frame = trace.GetFrame(i); MethodBase method = frame.GetMethod(); if (method == null) { continue; } Type type = method.DeclaringType; if (type != null) { TypeWrapper tw = ClassLoaderWrapper.GetWrapperFromType(type); if (tw != null) { ClassLoaderWrapper classLoader = tw.GetClassLoader(); AssemblyClassLoader acl = classLoader as AssemblyClassLoader; if (acl == null || acl.GetAssembly(tw) != typeof(object).Assembly) { java.lang.ClassLoader javaClassLoader = classLoader.GetJavaClassLoader(); if (javaClassLoader != null) { return javaClassLoader; } } } } } return null; } } static class Java_sun_misc_VMSupport { public static object initAgentProperties(object props) { return props; } }