diff src/share/vm/c1x/c1x_Compiler.cpp @ 1423:760213a60e8b

* rewrite of the code installation * partial support for safepoints * macro-based CiTargetMethod interface * code stub support
author Lukas Stadler <lukas.stadler@oracle.com>
date Mon, 16 Aug 2010 18:59:36 -0700
parents 3483ec571caf
children 98fffb304868
line wrap: on
line diff
--- a/src/share/vm/c1x/c1x_Compiler.cpp	Mon Aug 02 15:44:38 2010 -0700
+++ b/src/share/vm/c1x/c1x_Compiler.cpp	Mon Aug 16 18:59:36 2010 -0700
@@ -31,7 +31,7 @@
 void C1XCompiler::initialize() {
   if (_initialized) return;
   _initialized = true;
-	TRACE_C1X_1("initialize");
+    TRACE_C1X_1("initialize");
 
   JNIEnv *env = ((JavaThread *)Thread::current())->jni_environment();
   jclass klass = env->FindClass("com/sun/hotspot/c1x/VMEntriesNative");
@@ -39,6 +39,8 @@
   env->RegisterNatives(klass, VMEntries_methods, VMEntries_methods_count() );
   
   check_pending_exception("Could not register natives");
+
+  compute_offsets();
 }
 
 // Compilation entry point for methods
@@ -50,9 +52,14 @@
   ResourceMark rm;
   HandleMark hm;
 
+  C1XObjects::initializeObjects();
+
   CompilerThread::current()->set_compiling(true);
-  VMExits::compileMethod((methodOop)target->get_oop(), entry_bci);
+  methodOop method = (methodOop)target->get_oop();
+  VMExits::compileMethod(C1XObjects::add<methodOop>(method), C1XObjects::toString<Handle>(method->name(), THREAD), entry_bci);
   CompilerThread::current()->set_compiling(false);
+
+  C1XObjects::cleanupLocalObjects();
 }
 
 // Print compilation timers and statistics
@@ -60,129 +67,128 @@
 	TRACE_C1X_1("print_timers");
 }
 
-oop C1XCompiler::get_RiType(oop name, klassOop accessingType, TRAPS) {
-  symbolOop klass = java_lang_String::as_symbol_or_null(name);
-
-  if (klass == vmSymbols::byte_signature()) {
-    return VMExits::createRiTypePrimitive((int)T_BYTE, THREAD);
-  } else if (klass == vmSymbols::char_signature()) {
-    return VMExits::createRiTypePrimitive((int)T_CHAR, THREAD);
-  } else if (klass == vmSymbols::double_signature()) {
-    return VMExits::createRiTypePrimitive((int)T_DOUBLE, THREAD);
-  } else if (klass == vmSymbols::float_signature()) {
-    return VMExits::createRiTypePrimitive((int)T_FLOAT, THREAD);
-  } else if (klass == vmSymbols::int_signature()) {
-    return VMExits::createRiTypePrimitive((int)T_INT, THREAD);
-  } else if (klass == vmSymbols::long_signature()) {
-    return VMExits::createRiTypePrimitive((int)T_LONG, THREAD);
-  } else if (klass == vmSymbols::bool_signature()) {
-    return VMExits::createRiTypePrimitive((int)T_BOOLEAN, THREAD);
-  }
-  Handle classloader;
-  if (accessingType != NULL) {
-    classloader = accessingType->klass_part()->class_loader();
-  }
-  klassOop resolved_type = SystemDictionary::resolve_or_null(klass, classloader, accessingType->klass_part()->protection_domain(), Thread::current());
-  if (resolved_type != NULL) {
-    return VMExits::createRiType(resolved_type, THREAD);
-  } else {
-    return VMExits::createRiTypeUnresolved(klass, accessingType, THREAD);
-  }
-}
-
 oop C1XCompiler::get_RiType(ciType *type, klassOop accessor, TRAPS) {
   if (type->is_loaded()) {
     if (type->is_primitive_type()) {
       return VMExits::createRiTypePrimitive((int)type->basic_type(), THREAD);
     }
-    return VMExits::createRiType((klassOop)type->get_oop(), THREAD);
+    klassOop klass = (klassOop)type->get_oop();
+    return VMExits::createRiType(C1XObjects::add<klassOop>(klass), C1XObjects::toString<Handle>(klass->klass_part()->name(), THREAD), THREAD);
   } else {
-    return VMExits::createRiTypeUnresolved(((ciKlass *)type)->name()->get_symbolOop(), accessor, THREAD);
+    symbolOop name = ((ciKlass *)type)->name()->get_symbolOop();
+    return VMExits::createRiTypeUnresolved(C1XObjects::toString<Handle>(name, THREAD), C1XObjects::add<klassOop>(accessor), THREAD);
   }
 }
 
 oop C1XCompiler::get_RiField(ciField *field, TRAPS) {
   oop field_holder = get_RiType(field->holder(), NULL, CHECK_0);
   oop field_type = get_RiType(field->type(), NULL, CHECK_0);
-  symbolOop field_name = field->name()->get_symbolOop();
+  Handle field_name = C1XObjects::toString<Handle>(field->name()->get_symbolOop(), CHECK_0);
   int offset = field->offset();
 
   // TODO: implement caching
   return VMExits::createRiField(field_holder, field_name, field_type, offset, THREAD);
 }
 
-/*
-oop C1XCompiler::get_RiMethod(ciMethod *method, TRAPS) {
-  methodOop m = (methodOop)method->get_oop();
-  return get_RiMethod(m, THREAD);
+// C1XObjects implementation
+
+GrowableArray<address>* C1XObjects::_stubs = NULL;
+GrowableArray<jobject>* C1XObjects::_localHandles = NULL;
+
+void C1XObjects::initializeObjects() {
+  if (_stubs == NULL) {
+    assert(_localHandles == NULL, "inconsistent state");
+    _stubs = new(ResourceObj::C_HEAP) GrowableArray<address>(64, true);
+    _localHandles = new(ResourceObj::C_HEAP) GrowableArray<jobject>(64, true);
+  }
+  assert(_localHandles->length() == 0, "invalid state");
+}
+
+void C1XObjects::cleanupLocalObjects() {
+  for (int i=0; i<_localHandles->length(); i++) {
+    JNIHandles::destroy_global(_localHandles->at(i));
+  }
+  _localHandles->clear();
+}
+
+jlong C1XObjects::addStub(address stub) {
+  assert(!_stubs->contains(stub), "duplicate stub");
+  return _stubs->append(stub) | STUB;
 }
 
-oop C1XCompiler::get_RiMethod(methodOop m, TRAPS) {
-  // TODO: implement caching
-  return VMExits::createRiMethod(m, THREAD);
+jlong C1XObjects::add(Handle obj, CompilerObjectType type) {
+  assert(!obj.is_null(), "cannot add NULL handle");
+  int idx = -1;
+  for (int i=0; i<_localHandles->length(); i++)
+    if (JNIHandles::resolve_non_null(_localHandles->at(i)) == obj()) {
+      idx = i;
+      break;
+    }
+  if (idx = -1) {
+    if (JavaThread::current()->thread_state() == _thread_in_vm) {
+      idx = _localHandles->append(JNIHandles::make_global(obj));
+    } else {
+      VM_ENTRY_MARK;
+      idx = _localHandles->append(JNIHandles::make_global(obj));
+    }
+  }
+  return idx | type;
+}
+
+address C1XObjects::getStub(jlong id) {
+  assert((id & TYPE_MASK) == STUB, "wrong id type, STUB expected");
+  assert((id & ~TYPE_MASK) >= 0 && (id & ~TYPE_MASK) < _stubs->length(), "STUB index out of bounds");
+  return _stubs->at(id & ~TYPE_MASK);
+}
+
+oop C1XObjects::getObject(jlong id) {
+  assert((id & TYPE_MASK) != STUB, "wrong id type");
+  assert((id & ~TYPE_MASK) >= 0 && (id & ~TYPE_MASK) < _localHandles->length(), "index out of bounds");
+  return JNIHandles::resolve_non_null(_localHandles->at(id & ~TYPE_MASK));
 }
 
 
-oop C1XCompiler::get_RiType(klassOop klass, TRAPS) {
-  // TODO: implement caching
-  return VMExits::createRiType(klass, THREAD);
-}
-
-oop C1XCompiler::get_RiConstantPool(constantPoolOop cp, TRAPS) {
-  // TODO: implement caching
-  return VMExits::createRiConstantPool(cp, THREAD);
-}
+static void compute_offset(int &dest_offset, klassOop klass_oop, const char* name, const char* signature) {
+  symbolOop name_symbol = SymbolTable::probe(name, strlen(name));
+  symbolOop signature_symbol = SymbolTable::probe(signature, strlen(signature));
+  assert(name_symbol != NULL, "symbol not found - class layout changed?");
+  assert(signature_symbol != NULL, "symbol not found - class layout changed?");
 
-oop C1XCompiler::get_unresolved_RiType(symbolOop klass, klassOop accessingType, TRAPS)  {
-  // TODO: implement caching
-  return VMExits::createRiTypeUnresolved(klass, accessingType, THREAD);
-}
-*/
-
-// conversion internal objects -> reflected objects
-
-oop C1XObjects::getReflectedMethod(methodOop method, TRAPS) {
-  if (method->is_initializer()) {
-    return Reflection::new_constructor(method, CHECK_0);
-  } else {
-    return Reflection::new_method(method, UseNewReflection, false, CHECK_0);
+  instanceKlass* ik = instanceKlass::cast(klass_oop);
+  fieldDescriptor fd;
+  if (!ik->find_field(name_symbol, signature_symbol, &fd)) {
+    ResourceMark rm;
+    tty->print_cr("Invalid layout of %s at %s", ik->external_name(), name_symbol->as_C_string());
+    fatal("Invalid layout of preloaded class");
   }
-}
-
-oop C1XObjects::getReflectedClass(klassOop klass) {
-  return klass->klass_part()->java_mirror();
+  dest_offset = fd.offset();
 }
 
-oop C1XObjects::getReflectedSymbol(symbolOop symbol, TRAPS) {
-  return java_lang_String::create_from_symbol(symbol, THREAD)();
+// create the compute_class
+#define START_CLASS(name) { klassOop k = SystemDictionary::name##_klass();
+
+#define END_CLASS }
+
+#define FIELD(klass, name, signature) compute_offset(klass::_##name##_offset, k, #name, signature);
+#define CHAR_FIELD(klass, name) FIELD(klass, name, "C")
+#define INT_FIELD(klass, name) FIELD(klass, name, "I")
+#define LONG_FIELD(klass, name) FIELD(klass, name, "J")
+#define OOP_FIELD(klass, name, signature) FIELD(klass, name, signature)
+
+
+void C1XCompiler::compute_offsets() {
+  COMPILER_CLASSES_DO(START_CLASS, END_CLASS, CHAR_FIELD, INT_FIELD, LONG_FIELD, OOP_FIELD)
 }
 
-// conversion reflected objects -> internal objects
-
-methodOop C1XObjects::getInternalMethod(oop method) {
-  // copied from JNIEnv::FromReflectedMethod
-  oop mirror     = NULL;
-  int slot       = 0;
+#define EMPTY0
+#define EMPTY1(x)
+#define EMPTY2(x,y)
+#define FIELD2(klass, name) int klass::_##name##_offset = 0;
+#define FIELD3(klass, name, sig) FIELD2(klass, name)
 
-  if (method->klass() == SystemDictionary::reflect_Constructor_klass()) {
-    mirror = java_lang_reflect_Constructor::clazz(method);
-    slot   = java_lang_reflect_Constructor::slot(method);
-  } else {
-    assert(method->klass() == SystemDictionary::reflect_Method_klass(), "wrong type");
-    mirror = java_lang_reflect_Method::clazz(method);
-    slot   = java_lang_reflect_Method::slot(method);
-  }
-  klassOop k     = java_lang_Class::as_klassOop(mirror);
-  return instanceKlass::cast(k)->method_with_idnum(slot);
-}
-
-klassOop C1XObjects::getInternalClass(oop klass) {
-  return java_lang_Class::as_klassOop(klass);
-}
-
-symbolOop C1XObjects::getInternalSymbol(oop string) {
-  return java_lang_String::as_symbol_or_null(string);
-}
+COMPILER_CLASSES_DO(EMPTY1, EMPTY0, FIELD2, FIELD2, FIELD2, FIELD3)
 
 
 
+
+