Jo Shields 3c1f479b9d Imported Upstream version 4.0.0~alpha1
Former-commit-id: 806294f5ded97629b74c85c09952f2a74fe182d9
2015-04-07 09:35:12 +01:00

760 lines
21 KiB
C#

/*
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<int, long> handler = new Dictionary<int, long>();
// 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, "<Unknown>", 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<T>(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<Type, Atomic> impls = new Dictionary<Type, Atomic>();
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<Type, Atomic> curr = impls;
Dictionary<Type, Atomic> copy = new Dictionary<Type, Atomic>(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<T> : 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<T>(ref ((T[])array)[index], (T)value, (T)comparand);
}
}
}
static class Stats
{
#if !FIRST_PASS && UNSAFE_STATISTICS
private static readonly Dictionary<string, int> dict = new Dictionary<string, int>();
static Stats()
{
java.lang.Runtime.getRuntime().addShutdownHook(new DumpStats());
}
sealed class DumpStats : java.lang.Thread
{
public override void run()
{
List<KeyValuePair<string, int>> list = new List<KeyValuePair<string, int>>(dict);
list.Sort(delegate(KeyValuePair<string, int> kv1, KeyValuePair<string, int> kv2) { return kv1.Value.CompareTo(kv2.Value); });
foreach (KeyValuePair<string, int> 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;
}
}