Merge
This commit is contained in:
commit
b8c12c2e14
@ -219,3 +219,4 @@ cb51fb4789ac0b8be4056482077ddfb8f3bd3805 jdk8-b91
|
|||||||
785d07fe38901ecc1b7e0145e53e1c3da9361fee jdk8-b95
|
785d07fe38901ecc1b7e0145e53e1c3da9361fee jdk8-b95
|
||||||
c156084add486f941c12d886a0b1b2854795d557 jdk8-b96
|
c156084add486f941c12d886a0b1b2854795d557 jdk8-b96
|
||||||
a1c1e8bf71f354f3aec0214cf13d6668811e021d jdk8-b97
|
a1c1e8bf71f354f3aec0214cf13d6668811e021d jdk8-b97
|
||||||
|
0d0c983a817bbe8518a5ff201306334a8de267f2 jdk8-b98
|
||||||
|
@ -219,3 +219,4 @@ c8286839d0df04aba819ec4bef12b86babccf30e jdk8-b90
|
|||||||
2cf36f43df36137980d9828cec27003ec10daeee jdk8-b95
|
2cf36f43df36137980d9828cec27003ec10daeee jdk8-b95
|
||||||
3357c2776431d51a8de326a85e0f41420e40774f jdk8-b96
|
3357c2776431d51a8de326a85e0f41420e40774f jdk8-b96
|
||||||
469995a8e97424f450c880606d689bf345277b19 jdk8-b97
|
469995a8e97424f450c880606d689bf345277b19 jdk8-b97
|
||||||
|
3370fb6146e47a6cc05a213fc213e12fc0a38d07 jdk8-b98
|
||||||
|
@ -357,3 +357,5 @@ e6a4b8c71fa6f225bd989a34de2d0d0a656a8be8 jdk8-b96
|
|||||||
2b9380b0bf0b649f40704735773e8956c2d88ba0 hs25-b39
|
2b9380b0bf0b649f40704735773e8956c2d88ba0 hs25-b39
|
||||||
d197d377ab2e016d024e8c86cb06a57bd7eae590 jdk8-b97
|
d197d377ab2e016d024e8c86cb06a57bd7eae590 jdk8-b97
|
||||||
c9dd82da51ed34a28f7c6b3245163ee962e94572 hs25-b40
|
c9dd82da51ed34a28f7c6b3245163ee962e94572 hs25-b40
|
||||||
|
30b5b75c42ac5174b640fbef8aa87527668e8400 jdk8-b98
|
||||||
|
2b9946e10587f74ef75ae8145bea484df4a2738b hs25-b41
|
||||||
|
@ -49,7 +49,6 @@ public class ArrayKlass extends Klass {
|
|||||||
higherDimension = new MetadataField(type.getAddressField("_higher_dimension"), 0);
|
higherDimension = new MetadataField(type.getAddressField("_higher_dimension"), 0);
|
||||||
lowerDimension = new MetadataField(type.getAddressField("_lower_dimension"), 0);
|
lowerDimension = new MetadataField(type.getAddressField("_lower_dimension"), 0);
|
||||||
vtableLen = new CIntField(type.getCIntegerField("_vtable_len"), 0);
|
vtableLen = new CIntField(type.getCIntegerField("_vtable_len"), 0);
|
||||||
allocSize = new CIntField(type.getCIntegerField("_alloc_size"), 0);
|
|
||||||
componentMirror = new OopField(type.getOopField("_component_mirror"), 0);
|
componentMirror = new OopField(type.getOopField("_component_mirror"), 0);
|
||||||
javaLangCloneableName = null;
|
javaLangCloneableName = null;
|
||||||
javaLangObjectName = null;
|
javaLangObjectName = null;
|
||||||
@ -64,7 +63,6 @@ public class ArrayKlass extends Klass {
|
|||||||
private static MetadataField higherDimension;
|
private static MetadataField higherDimension;
|
||||||
private static MetadataField lowerDimension;
|
private static MetadataField lowerDimension;
|
||||||
private static CIntField vtableLen;
|
private static CIntField vtableLen;
|
||||||
private static CIntField allocSize;
|
|
||||||
private static OopField componentMirror;
|
private static OopField componentMirror;
|
||||||
|
|
||||||
public Klass getJavaSuper() {
|
public Klass getJavaSuper() {
|
||||||
@ -76,7 +74,6 @@ public class ArrayKlass extends Klass {
|
|||||||
public Klass getHigherDimension() { return (Klass) higherDimension.getValue(this); }
|
public Klass getHigherDimension() { return (Klass) higherDimension.getValue(this); }
|
||||||
public Klass getLowerDimension() { return (Klass) lowerDimension.getValue(this); }
|
public Klass getLowerDimension() { return (Klass) lowerDimension.getValue(this); }
|
||||||
public long getVtableLen() { return vtableLen.getValue(this); }
|
public long getVtableLen() { return vtableLen.getValue(this); }
|
||||||
public long getAllocSize() { return allocSize.getValue(this); }
|
|
||||||
public Oop getComponentMirror() { return componentMirror.getValue(this); }
|
public Oop getComponentMirror() { return componentMirror.getValue(this); }
|
||||||
|
|
||||||
// constant class names - javaLangCloneable, javaIoSerializable, javaLangObject
|
// constant class names - javaLangCloneable, javaIoSerializable, javaLangObject
|
||||||
@ -147,7 +144,6 @@ public class ArrayKlass extends Klass {
|
|||||||
visitor.doMetadata(higherDimension, true);
|
visitor.doMetadata(higherDimension, true);
|
||||||
visitor.doMetadata(lowerDimension, true);
|
visitor.doMetadata(lowerDimension, true);
|
||||||
visitor.doCInt(vtableLen, true);
|
visitor.doCInt(vtableLen, true);
|
||||||
visitor.doCInt(allocSize, true);
|
|
||||||
visitor.doOop(componentMirror, true);
|
visitor.doOop(componentMirror, true);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -57,7 +57,6 @@ public class Klass extends Metadata implements ClassConstants {
|
|||||||
accessFlags = new CIntField(type.getCIntegerField("_access_flags"), 0);
|
accessFlags = new CIntField(type.getCIntegerField("_access_flags"), 0);
|
||||||
subklass = new MetadataField(type.getAddressField("_subklass"), 0);
|
subklass = new MetadataField(type.getAddressField("_subklass"), 0);
|
||||||
nextSibling = new MetadataField(type.getAddressField("_next_sibling"), 0);
|
nextSibling = new MetadataField(type.getAddressField("_next_sibling"), 0);
|
||||||
allocCount = new CIntField(type.getCIntegerField("_alloc_count"), 0);
|
|
||||||
|
|
||||||
LH_INSTANCE_SLOW_PATH_BIT = db.lookupIntConstant("Klass::_lh_instance_slow_path_bit").intValue();
|
LH_INSTANCE_SLOW_PATH_BIT = db.lookupIntConstant("Klass::_lh_instance_slow_path_bit").intValue();
|
||||||
LH_LOG2_ELEMENT_SIZE_SHIFT = db.lookupIntConstant("Klass::_lh_log2_element_size_shift").intValue();
|
LH_LOG2_ELEMENT_SIZE_SHIFT = db.lookupIntConstant("Klass::_lh_log2_element_size_shift").intValue();
|
||||||
@ -87,7 +86,6 @@ public class Klass extends Metadata implements ClassConstants {
|
|||||||
private static CIntField accessFlags;
|
private static CIntField accessFlags;
|
||||||
private static MetadataField subklass;
|
private static MetadataField subklass;
|
||||||
private static MetadataField nextSibling;
|
private static MetadataField nextSibling;
|
||||||
private static CIntField allocCount;
|
|
||||||
|
|
||||||
private Address getValue(AddressField field) {
|
private Address getValue(AddressField field) {
|
||||||
return addr.getAddressAt(field.getOffset());
|
return addr.getAddressAt(field.getOffset());
|
||||||
@ -108,7 +106,6 @@ public class Klass extends Metadata implements ClassConstants {
|
|||||||
public AccessFlags getAccessFlagsObj(){ return new AccessFlags(getAccessFlags()); }
|
public AccessFlags getAccessFlagsObj(){ return new AccessFlags(getAccessFlags()); }
|
||||||
public Klass getSubklassKlass() { return (Klass) subklass.getValue(this); }
|
public Klass getSubklassKlass() { return (Klass) subklass.getValue(this); }
|
||||||
public Klass getNextSiblingKlass() { return (Klass) nextSibling.getValue(this); }
|
public Klass getNextSiblingKlass() { return (Klass) nextSibling.getValue(this); }
|
||||||
public long getAllocCount() { return allocCount.getValue(this); }
|
|
||||||
|
|
||||||
// computed access flags - takes care of inner classes etc.
|
// computed access flags - takes care of inner classes etc.
|
||||||
// This is closer to actual source level than getAccessFlags() etc.
|
// This is closer to actual source level than getAccessFlags() etc.
|
||||||
@ -172,7 +169,6 @@ public class Klass extends Metadata implements ClassConstants {
|
|||||||
visitor.doCInt(accessFlags, true);
|
visitor.doCInt(accessFlags, true);
|
||||||
visitor.doMetadata(subklass, true);
|
visitor.doMetadata(subklass, true);
|
||||||
visitor.doMetadata(nextSibling, true);
|
visitor.doMetadata(nextSibling, true);
|
||||||
visitor.doCInt(allocCount, true);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public long getObjectSize() {
|
public long getObjectSize() {
|
||||||
|
@ -221,7 +221,6 @@
|
|||||||
_JVM_SetLength
|
_JVM_SetLength
|
||||||
_JVM_SetNativeThreadName
|
_JVM_SetNativeThreadName
|
||||||
_JVM_SetPrimitiveArrayElement
|
_JVM_SetPrimitiveArrayElement
|
||||||
_JVM_SetProtectionDomain
|
|
||||||
_JVM_SetSockOpt
|
_JVM_SetSockOpt
|
||||||
_JVM_SetThreadPriority
|
_JVM_SetThreadPriority
|
||||||
_JVM_Sleep
|
_JVM_Sleep
|
||||||
|
@ -221,7 +221,6 @@
|
|||||||
_JVM_SetLength
|
_JVM_SetLength
|
||||||
_JVM_SetNativeThreadName
|
_JVM_SetNativeThreadName
|
||||||
_JVM_SetPrimitiveArrayElement
|
_JVM_SetPrimitiveArrayElement
|
||||||
_JVM_SetProtectionDomain
|
|
||||||
_JVM_SetSockOpt
|
_JVM_SetSockOpt
|
||||||
_JVM_SetThreadPriority
|
_JVM_SetThreadPriority
|
||||||
_JVM_Sleep
|
_JVM_Sleep
|
||||||
|
@ -35,7 +35,7 @@ HOTSPOT_VM_COPYRIGHT=Copyright 2013
|
|||||||
|
|
||||||
HS_MAJOR_VER=25
|
HS_MAJOR_VER=25
|
||||||
HS_MINOR_VER=0
|
HS_MINOR_VER=0
|
||||||
HS_BUILD_NUMBER=40
|
HS_BUILD_NUMBER=41
|
||||||
|
|
||||||
JDK_MAJOR_VER=1
|
JDK_MAJOR_VER=1
|
||||||
JDK_MINOR_VER=8
|
JDK_MINOR_VER=8
|
||||||
|
@ -223,7 +223,6 @@ SUNWprivate_1.1 {
|
|||||||
JVM_SetLength;
|
JVM_SetLength;
|
||||||
JVM_SetNativeThreadName;
|
JVM_SetNativeThreadName;
|
||||||
JVM_SetPrimitiveArrayElement;
|
JVM_SetPrimitiveArrayElement;
|
||||||
JVM_SetProtectionDomain;
|
|
||||||
JVM_SetSockOpt;
|
JVM_SetSockOpt;
|
||||||
JVM_SetThreadPriority;
|
JVM_SetThreadPriority;
|
||||||
JVM_Sleep;
|
JVM_Sleep;
|
||||||
|
@ -223,7 +223,6 @@ SUNWprivate_1.1 {
|
|||||||
JVM_SetLength;
|
JVM_SetLength;
|
||||||
JVM_SetNativeThreadName;
|
JVM_SetNativeThreadName;
|
||||||
JVM_SetPrimitiveArrayElement;
|
JVM_SetPrimitiveArrayElement;
|
||||||
JVM_SetProtectionDomain;
|
|
||||||
JVM_SetSockOpt;
|
JVM_SetSockOpt;
|
||||||
JVM_SetThreadPriority;
|
JVM_SetThreadPriority;
|
||||||
JVM_Sleep;
|
JVM_Sleep;
|
||||||
|
@ -223,7 +223,6 @@ SUNWprivate_1.1 {
|
|||||||
JVM_SetLength;
|
JVM_SetLength;
|
||||||
JVM_SetNativeThreadName;
|
JVM_SetNativeThreadName;
|
||||||
JVM_SetPrimitiveArrayElement;
|
JVM_SetPrimitiveArrayElement;
|
||||||
JVM_SetProtectionDomain;
|
|
||||||
JVM_SetSockOpt;
|
JVM_SetSockOpt;
|
||||||
JVM_SetThreadPriority;
|
JVM_SetThreadPriority;
|
||||||
JVM_Sleep;
|
JVM_Sleep;
|
||||||
|
@ -1234,12 +1234,13 @@ bool os::address_is_in_vm(address addr) {
|
|||||||
Dl_info dlinfo;
|
Dl_info dlinfo;
|
||||||
|
|
||||||
if (libjvm_base_addr == NULL) {
|
if (libjvm_base_addr == NULL) {
|
||||||
dladdr(CAST_FROM_FN_PTR(void *, os::address_is_in_vm), &dlinfo);
|
if (dladdr(CAST_FROM_FN_PTR(void *, os::address_is_in_vm), &dlinfo) != 0) {
|
||||||
libjvm_base_addr = (address)dlinfo.dli_fbase;
|
libjvm_base_addr = (address)dlinfo.dli_fbase;
|
||||||
|
}
|
||||||
assert(libjvm_base_addr !=NULL, "Cannot obtain base address for libjvm");
|
assert(libjvm_base_addr !=NULL, "Cannot obtain base address for libjvm");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dladdr((void *)addr, &dlinfo)) {
|
if (dladdr((void *)addr, &dlinfo) != 0) {
|
||||||
if (libjvm_base_addr == (address)dlinfo.dli_fbase) return true;
|
if (libjvm_base_addr == (address)dlinfo.dli_fbase) return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1251,35 +1252,40 @@ bool os::address_is_in_vm(address addr) {
|
|||||||
|
|
||||||
bool os::dll_address_to_function_name(address addr, char *buf,
|
bool os::dll_address_to_function_name(address addr, char *buf,
|
||||||
int buflen, int *offset) {
|
int buflen, int *offset) {
|
||||||
|
// buf is not optional, but offset is optional
|
||||||
|
assert(buf != NULL, "sanity check");
|
||||||
|
|
||||||
Dl_info dlinfo;
|
Dl_info dlinfo;
|
||||||
char localbuf[MACH_MAXSYMLEN];
|
char localbuf[MACH_MAXSYMLEN];
|
||||||
|
|
||||||
// dladdr will find names of dynamic functions only, but does
|
if (dladdr((void*)addr, &dlinfo) != 0) {
|
||||||
// it set dli_fbase with mach_header address when it "fails" ?
|
// see if we have a matching symbol
|
||||||
if (dladdr((void*)addr, &dlinfo) && dlinfo.dli_sname != NULL) {
|
if (dlinfo.dli_saddr != NULL && dlinfo.dli_sname != NULL) {
|
||||||
if (buf != NULL) {
|
if (!Decoder::demangle(dlinfo.dli_sname, buf, buflen)) {
|
||||||
if(!Decoder::demangle(dlinfo.dli_sname, buf, buflen)) {
|
|
||||||
jio_snprintf(buf, buflen, "%s", dlinfo.dli_sname);
|
jio_snprintf(buf, buflen, "%s", dlinfo.dli_sname);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
if (offset != NULL) *offset = addr - (address)dlinfo.dli_saddr;
|
if (offset != NULL) *offset = addr - (address)dlinfo.dli_saddr;
|
||||||
return true;
|
return true;
|
||||||
} else if (dlinfo.dli_fname != NULL && dlinfo.dli_fbase != 0) {
|
}
|
||||||
|
// no matching symbol so try for just file info
|
||||||
|
if (dlinfo.dli_fname != NULL && dlinfo.dli_fbase != NULL) {
|
||||||
if (Decoder::decode((address)(addr - (address)dlinfo.dli_fbase),
|
if (Decoder::decode((address)(addr - (address)dlinfo.dli_fbase),
|
||||||
buf, buflen, offset, dlinfo.dli_fname)) {
|
buf, buflen, offset, dlinfo.dli_fname)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Handle non-dymanic manually:
|
// Handle non-dynamic manually:
|
||||||
if (dlinfo.dli_fbase != NULL &&
|
if (dlinfo.dli_fbase != NULL &&
|
||||||
Decoder::decode(addr, localbuf, MACH_MAXSYMLEN, offset, dlinfo.dli_fbase)) {
|
Decoder::decode(addr, localbuf, MACH_MAXSYMLEN, offset,
|
||||||
if(!Decoder::demangle(localbuf, buf, buflen)) {
|
dlinfo.dli_fbase)) {
|
||||||
|
if (!Decoder::demangle(localbuf, buf, buflen)) {
|
||||||
jio_snprintf(buf, buflen, "%s", localbuf);
|
jio_snprintf(buf, buflen, "%s", localbuf);
|
||||||
}
|
}
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (buf != NULL) buf[0] = '\0';
|
}
|
||||||
|
buf[0] = '\0';
|
||||||
if (offset != NULL) *offset = -1;
|
if (offset != NULL) *offset = -1;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -1287,17 +1293,24 @@ bool os::dll_address_to_function_name(address addr, char *buf,
|
|||||||
// ported from solaris version
|
// ported from solaris version
|
||||||
bool os::dll_address_to_library_name(address addr, char* buf,
|
bool os::dll_address_to_library_name(address addr, char* buf,
|
||||||
int buflen, int* offset) {
|
int buflen, int* offset) {
|
||||||
|
// buf is not optional, but offset is optional
|
||||||
|
assert(buf != NULL, "sanity check");
|
||||||
|
|
||||||
Dl_info dlinfo;
|
Dl_info dlinfo;
|
||||||
|
|
||||||
if (dladdr((void*)addr, &dlinfo)){
|
if (dladdr((void*)addr, &dlinfo) != 0) {
|
||||||
if (buf) jio_snprintf(buf, buflen, "%s", dlinfo.dli_fname);
|
if (dlinfo.dli_fname != NULL) {
|
||||||
if (offset) *offset = addr - (address)dlinfo.dli_fbase;
|
jio_snprintf(buf, buflen, "%s", dlinfo.dli_fname);
|
||||||
|
}
|
||||||
|
if (dlinfo.dli_fbase != NULL && offset != NULL) {
|
||||||
|
*offset = addr - (address)dlinfo.dli_fbase;
|
||||||
|
}
|
||||||
return true;
|
return true;
|
||||||
} else {
|
}
|
||||||
if (buf) buf[0] = '\0';
|
|
||||||
|
buf[0] = '\0';
|
||||||
if (offset) *offset = -1;
|
if (offset) *offset = -1;
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Loads .dll/.so and
|
// Loads .dll/.so and
|
||||||
@ -1527,7 +1540,8 @@ void os::print_dll_info(outputStream *st) {
|
|||||||
Link_map *map;
|
Link_map *map;
|
||||||
Link_map *p;
|
Link_map *p;
|
||||||
|
|
||||||
if (!dladdr(CAST_FROM_FN_PTR(void *, os::print_dll_info), &dli)) {
|
if (dladdr(CAST_FROM_FN_PTR(void *, os::print_dll_info), &dli) == 0 ||
|
||||||
|
dli.dli_fname == NULL) {
|
||||||
st->print_cr("Error: Cannot print dynamic libraries.");
|
st->print_cr("Error: Cannot print dynamic libraries.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -1707,8 +1721,11 @@ void os::jvm_path(char *buf, jint buflen) {
|
|||||||
bool ret = dll_address_to_library_name(
|
bool ret = dll_address_to_library_name(
|
||||||
CAST_FROM_FN_PTR(address, os::jvm_path),
|
CAST_FROM_FN_PTR(address, os::jvm_path),
|
||||||
dli_fname, sizeof(dli_fname), NULL);
|
dli_fname, sizeof(dli_fname), NULL);
|
||||||
assert(ret != 0, "cannot locate libjvm");
|
assert(ret, "cannot locate libjvm");
|
||||||
char *rp = realpath(dli_fname, buf);
|
char *rp = NULL;
|
||||||
|
if (ret && dli_fname[0] != '\0') {
|
||||||
|
rp = realpath(dli_fname, buf);
|
||||||
|
}
|
||||||
if (rp == NULL)
|
if (rp == NULL)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -3747,20 +3764,20 @@ int os::Bsd::safe_cond_timedwait(pthread_cond_t *_cond, pthread_mutex_t *_mutex,
|
|||||||
bool os::find(address addr, outputStream* st) {
|
bool os::find(address addr, outputStream* st) {
|
||||||
Dl_info dlinfo;
|
Dl_info dlinfo;
|
||||||
memset(&dlinfo, 0, sizeof(dlinfo));
|
memset(&dlinfo, 0, sizeof(dlinfo));
|
||||||
if (dladdr(addr, &dlinfo)) {
|
if (dladdr(addr, &dlinfo) != 0) {
|
||||||
st->print(PTR_FORMAT ": ", addr);
|
st->print(PTR_FORMAT ": ", addr);
|
||||||
if (dlinfo.dli_sname != NULL) {
|
if (dlinfo.dli_sname != NULL && dlinfo.dli_saddr != NULL) {
|
||||||
st->print("%s+%#x", dlinfo.dli_sname,
|
st->print("%s+%#x", dlinfo.dli_sname,
|
||||||
addr - (intptr_t)dlinfo.dli_saddr);
|
addr - (intptr_t)dlinfo.dli_saddr);
|
||||||
} else if (dlinfo.dli_fname) {
|
} else if (dlinfo.dli_fbase != NULL) {
|
||||||
st->print("<offset %#x>", addr - (intptr_t)dlinfo.dli_fbase);
|
st->print("<offset %#x>", addr - (intptr_t)dlinfo.dli_fbase);
|
||||||
} else {
|
} else {
|
||||||
st->print("<absolute address>");
|
st->print("<absolute address>");
|
||||||
}
|
}
|
||||||
if (dlinfo.dli_fname) {
|
if (dlinfo.dli_fname != NULL) {
|
||||||
st->print(" in %s", dlinfo.dli_fname);
|
st->print(" in %s", dlinfo.dli_fname);
|
||||||
}
|
}
|
||||||
if (dlinfo.dli_fbase) {
|
if (dlinfo.dli_fbase != NULL) {
|
||||||
st->print(" at " PTR_FORMAT, dlinfo.dli_fbase);
|
st->print(" at " PTR_FORMAT, dlinfo.dli_fbase);
|
||||||
}
|
}
|
||||||
st->cr();
|
st->cr();
|
||||||
@ -3773,7 +3790,7 @@ bool os::find(address addr, outputStream* st) {
|
|||||||
if (!lowest) lowest = (address) dlinfo.dli_fbase;
|
if (!lowest) lowest = (address) dlinfo.dli_fbase;
|
||||||
if (begin < lowest) begin = lowest;
|
if (begin < lowest) begin = lowest;
|
||||||
Dl_info dlinfo2;
|
Dl_info dlinfo2;
|
||||||
if (dladdr(end, &dlinfo2) && dlinfo2.dli_saddr != dlinfo.dli_saddr
|
if (dladdr(end, &dlinfo2) != 0 && dlinfo2.dli_saddr != dlinfo.dli_saddr
|
||||||
&& end > dlinfo2.dli_saddr && dlinfo2.dli_saddr > begin)
|
&& end > dlinfo2.dli_saddr && dlinfo2.dli_saddr > begin)
|
||||||
end = (address) dlinfo2.dli_saddr;
|
end = (address) dlinfo2.dli_saddr;
|
||||||
Disassembler::decode(begin, end, st);
|
Disassembler::decode(begin, end, st);
|
||||||
|
@ -1682,12 +1682,13 @@ bool os::address_is_in_vm(address addr) {
|
|||||||
Dl_info dlinfo;
|
Dl_info dlinfo;
|
||||||
|
|
||||||
if (libjvm_base_addr == NULL) {
|
if (libjvm_base_addr == NULL) {
|
||||||
dladdr(CAST_FROM_FN_PTR(void *, os::address_is_in_vm), &dlinfo);
|
if (dladdr(CAST_FROM_FN_PTR(void *, os::address_is_in_vm), &dlinfo) != 0) {
|
||||||
libjvm_base_addr = (address)dlinfo.dli_fbase;
|
libjvm_base_addr = (address)dlinfo.dli_fbase;
|
||||||
|
}
|
||||||
assert(libjvm_base_addr !=NULL, "Cannot obtain base address for libjvm");
|
assert(libjvm_base_addr !=NULL, "Cannot obtain base address for libjvm");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dladdr((void *)addr, &dlinfo)) {
|
if (dladdr((void *)addr, &dlinfo) != 0) {
|
||||||
if (libjvm_base_addr == (address)dlinfo.dli_fbase) return true;
|
if (libjvm_base_addr == (address)dlinfo.dli_fbase) return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1696,24 +1697,30 @@ bool os::address_is_in_vm(address addr) {
|
|||||||
|
|
||||||
bool os::dll_address_to_function_name(address addr, char *buf,
|
bool os::dll_address_to_function_name(address addr, char *buf,
|
||||||
int buflen, int *offset) {
|
int buflen, int *offset) {
|
||||||
|
// buf is not optional, but offset is optional
|
||||||
|
assert(buf != NULL, "sanity check");
|
||||||
|
|
||||||
Dl_info dlinfo;
|
Dl_info dlinfo;
|
||||||
|
|
||||||
if (dladdr((void*)addr, &dlinfo) && dlinfo.dli_sname != NULL) {
|
if (dladdr((void*)addr, &dlinfo) != 0) {
|
||||||
if (buf != NULL) {
|
// see if we have a matching symbol
|
||||||
if(!Decoder::demangle(dlinfo.dli_sname, buf, buflen)) {
|
if (dlinfo.dli_saddr != NULL && dlinfo.dli_sname != NULL) {
|
||||||
|
if (!Decoder::demangle(dlinfo.dli_sname, buf, buflen)) {
|
||||||
jio_snprintf(buf, buflen, "%s", dlinfo.dli_sname);
|
jio_snprintf(buf, buflen, "%s", dlinfo.dli_sname);
|
||||||
}
|
}
|
||||||
}
|
|
||||||
if (offset != NULL) *offset = addr - (address)dlinfo.dli_saddr;
|
if (offset != NULL) *offset = addr - (address)dlinfo.dli_saddr;
|
||||||
return true;
|
return true;
|
||||||
} else if (dlinfo.dli_fname != NULL && dlinfo.dli_fbase != 0) {
|
}
|
||||||
|
// no matching symbol so try for just file info
|
||||||
|
if (dlinfo.dli_fname != NULL && dlinfo.dli_fbase != NULL) {
|
||||||
if (Decoder::decode((address)(addr - (address)dlinfo.dli_fbase),
|
if (Decoder::decode((address)(addr - (address)dlinfo.dli_fbase),
|
||||||
buf, buflen, offset, dlinfo.dli_fname)) {
|
buf, buflen, offset, dlinfo.dli_fname)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if (buf != NULL) buf[0] = '\0';
|
buf[0] = '\0';
|
||||||
if (offset != NULL) *offset = -1;
|
if (offset != NULL) *offset = -1;
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -1764,6 +1771,9 @@ static int address_to_library_name_callback(struct dl_phdr_info *info,
|
|||||||
|
|
||||||
bool os::dll_address_to_library_name(address addr, char* buf,
|
bool os::dll_address_to_library_name(address addr, char* buf,
|
||||||
int buflen, int* offset) {
|
int buflen, int* offset) {
|
||||||
|
// buf is not optional, but offset is optional
|
||||||
|
assert(buf != NULL, "sanity check");
|
||||||
|
|
||||||
Dl_info dlinfo;
|
Dl_info dlinfo;
|
||||||
struct _address_to_library_name data;
|
struct _address_to_library_name data;
|
||||||
|
|
||||||
@ -1782,15 +1792,20 @@ bool os::dll_address_to_library_name(address addr, char* buf,
|
|||||||
// buf already contains library name
|
// buf already contains library name
|
||||||
if (offset) *offset = addr - data.base;
|
if (offset) *offset = addr - data.base;
|
||||||
return true;
|
return true;
|
||||||
} else if (dladdr((void*)addr, &dlinfo)){
|
}
|
||||||
if (buf) jio_snprintf(buf, buflen, "%s", dlinfo.dli_fname);
|
if (dladdr((void*)addr, &dlinfo) != 0) {
|
||||||
if (offset) *offset = addr - (address)dlinfo.dli_fbase;
|
if (dlinfo.dli_fname != NULL) {
|
||||||
|
jio_snprintf(buf, buflen, "%s", dlinfo.dli_fname);
|
||||||
|
}
|
||||||
|
if (dlinfo.dli_fbase != NULL && offset != NULL) {
|
||||||
|
*offset = addr - (address)dlinfo.dli_fbase;
|
||||||
|
}
|
||||||
return true;
|
return true;
|
||||||
} else {
|
}
|
||||||
if (buf) buf[0] = '\0';
|
|
||||||
|
buf[0] = '\0';
|
||||||
if (offset) *offset = -1;
|
if (offset) *offset = -1;
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Loads .dll/.so and
|
// Loads .dll/.so and
|
||||||
@ -2317,8 +2332,11 @@ void os::jvm_path(char *buf, jint buflen) {
|
|||||||
bool ret = dll_address_to_library_name(
|
bool ret = dll_address_to_library_name(
|
||||||
CAST_FROM_FN_PTR(address, os::jvm_path),
|
CAST_FROM_FN_PTR(address, os::jvm_path),
|
||||||
dli_fname, sizeof(dli_fname), NULL);
|
dli_fname, sizeof(dli_fname), NULL);
|
||||||
assert(ret != 0, "cannot locate libjvm");
|
assert(ret, "cannot locate libjvm");
|
||||||
char *rp = realpath(dli_fname, buf);
|
char *rp = NULL;
|
||||||
|
if (ret && dli_fname[0] != '\0') {
|
||||||
|
rp = realpath(dli_fname, buf);
|
||||||
|
}
|
||||||
if (rp == NULL)
|
if (rp == NULL)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
@ -4730,20 +4748,20 @@ int os::Linux::safe_cond_timedwait(pthread_cond_t *_cond, pthread_mutex_t *_mute
|
|||||||
bool os::find(address addr, outputStream* st) {
|
bool os::find(address addr, outputStream* st) {
|
||||||
Dl_info dlinfo;
|
Dl_info dlinfo;
|
||||||
memset(&dlinfo, 0, sizeof(dlinfo));
|
memset(&dlinfo, 0, sizeof(dlinfo));
|
||||||
if (dladdr(addr, &dlinfo)) {
|
if (dladdr(addr, &dlinfo) != 0) {
|
||||||
st->print(PTR_FORMAT ": ", addr);
|
st->print(PTR_FORMAT ": ", addr);
|
||||||
if (dlinfo.dli_sname != NULL) {
|
if (dlinfo.dli_sname != NULL && dlinfo.dli_saddr != NULL) {
|
||||||
st->print("%s+%#x", dlinfo.dli_sname,
|
st->print("%s+%#x", dlinfo.dli_sname,
|
||||||
addr - (intptr_t)dlinfo.dli_saddr);
|
addr - (intptr_t)dlinfo.dli_saddr);
|
||||||
} else if (dlinfo.dli_fname) {
|
} else if (dlinfo.dli_fbase != NULL) {
|
||||||
st->print("<offset %#x>", addr - (intptr_t)dlinfo.dli_fbase);
|
st->print("<offset %#x>", addr - (intptr_t)dlinfo.dli_fbase);
|
||||||
} else {
|
} else {
|
||||||
st->print("<absolute address>");
|
st->print("<absolute address>");
|
||||||
}
|
}
|
||||||
if (dlinfo.dli_fname) {
|
if (dlinfo.dli_fname != NULL) {
|
||||||
st->print(" in %s", dlinfo.dli_fname);
|
st->print(" in %s", dlinfo.dli_fname);
|
||||||
}
|
}
|
||||||
if (dlinfo.dli_fbase) {
|
if (dlinfo.dli_fbase != NULL) {
|
||||||
st->print(" at " PTR_FORMAT, dlinfo.dli_fbase);
|
st->print(" at " PTR_FORMAT, dlinfo.dli_fbase);
|
||||||
}
|
}
|
||||||
st->cr();
|
st->cr();
|
||||||
@ -4756,7 +4774,7 @@ bool os::find(address addr, outputStream* st) {
|
|||||||
if (!lowest) lowest = (address) dlinfo.dli_fbase;
|
if (!lowest) lowest = (address) dlinfo.dli_fbase;
|
||||||
if (begin < lowest) begin = lowest;
|
if (begin < lowest) begin = lowest;
|
||||||
Dl_info dlinfo2;
|
Dl_info dlinfo2;
|
||||||
if (dladdr(end, &dlinfo2) && dlinfo2.dli_saddr != dlinfo.dli_saddr
|
if (dladdr(end, &dlinfo2) != 0 && dlinfo2.dli_saddr != dlinfo.dli_saddr
|
||||||
&& end > dlinfo2.dli_saddr && dlinfo2.dli_saddr > begin)
|
&& end > dlinfo2.dli_saddr && dlinfo2.dli_saddr > begin)
|
||||||
end = (address) dlinfo2.dli_saddr;
|
end = (address) dlinfo2.dli_saddr;
|
||||||
Disassembler::decode(begin, end, st);
|
Disassembler::decode(begin, end, st);
|
||||||
|
@ -30,15 +30,6 @@
|
|||||||
//
|
//
|
||||||
#define RUNTIME_OS_FLAGS(develop, develop_pd, product, product_pd, diagnostic, notproduct) \
|
#define RUNTIME_OS_FLAGS(develop, develop_pd, product, product_pd, diagnostic, notproduct) \
|
||||||
\
|
\
|
||||||
product(bool, UseISM, false, \
|
|
||||||
"Use Intimate Shared Memory (Solaris Only)") \
|
|
||||||
\
|
|
||||||
product(bool, UsePermISM, false, \
|
|
||||||
"Obsolete flag for compatibility (same as UseISM)") \
|
|
||||||
\
|
|
||||||
product(bool, UseMPSS, true, \
|
|
||||||
"Use Multiple Page Size Support (Solaris 9 Only)") \
|
|
||||||
\
|
|
||||||
product(bool, UseExtendedFileIO, true, \
|
product(bool, UseExtendedFileIO, true, \
|
||||||
"Enable workaround for limitations of stdio FILE structure")
|
"Enable workaround for limitations of stdio FILE structure")
|
||||||
|
|
||||||
|
@ -115,45 +115,6 @@
|
|||||||
// for timer info max values which include all bits
|
// for timer info max values which include all bits
|
||||||
#define ALL_64_BITS CONST64(0xFFFFFFFFFFFFFFFF)
|
#define ALL_64_BITS CONST64(0xFFFFFFFFFFFFFFFF)
|
||||||
|
|
||||||
#ifdef _GNU_SOURCE
|
|
||||||
// See bug #6514594
|
|
||||||
extern "C" int madvise(caddr_t, size_t, int);
|
|
||||||
extern "C" int memcntl(caddr_t addr, size_t len, int cmd, caddr_t arg,
|
|
||||||
int attr, int mask);
|
|
||||||
#endif //_GNU_SOURCE
|
|
||||||
|
|
||||||
/*
|
|
||||||
MPSS Changes Start.
|
|
||||||
The JVM binary needs to be built and run on pre-Solaris 9
|
|
||||||
systems, but the constants needed by MPSS are only in Solaris 9
|
|
||||||
header files. They are textually replicated here to allow
|
|
||||||
building on earlier systems. Once building on Solaris 8 is
|
|
||||||
no longer a requirement, these #defines can be replaced by ordinary
|
|
||||||
system .h inclusion.
|
|
||||||
|
|
||||||
In earlier versions of the JDK and Solaris, we used ISM for large pages.
|
|
||||||
But ISM requires shared memory to achieve this and thus has many caveats.
|
|
||||||
MPSS is a fully transparent and is a cleaner way to get large pages.
|
|
||||||
Although we still require keeping ISM for backward compatiblitiy as well as
|
|
||||||
giving the opportunity to use large pages on older systems it is
|
|
||||||
recommended that MPSS be used for Solaris 9 and above.
|
|
||||||
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef MC_HAT_ADVISE
|
|
||||||
|
|
||||||
struct memcntl_mha {
|
|
||||||
uint_t mha_cmd; /* command(s) */
|
|
||||||
uint_t mha_flags;
|
|
||||||
size_t mha_pagesize;
|
|
||||||
};
|
|
||||||
#define MC_HAT_ADVISE 7 /* advise hat map size */
|
|
||||||
#define MHA_MAPSIZE_VA 0x1 /* set preferred page size */
|
|
||||||
#define MAP_ALIGN 0x200 /* addr specifies alignment */
|
|
||||||
|
|
||||||
#endif
|
|
||||||
// MPSS Changes End.
|
|
||||||
|
|
||||||
|
|
||||||
// Here are some liblgrp types from sys/lgrp_user.h to be able to
|
// Here are some liblgrp types from sys/lgrp_user.h to be able to
|
||||||
// compile on older systems without this header file.
|
// compile on older systems without this header file.
|
||||||
@ -172,32 +133,6 @@ struct memcntl_mha {
|
|||||||
# define LGRP_RSRC_MEM 1 /* memory resources */
|
# define LGRP_RSRC_MEM 1 /* memory resources */
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Some more macros from sys/mman.h that are not present in Solaris 8.
|
|
||||||
|
|
||||||
#ifndef MAX_MEMINFO_CNT
|
|
||||||
/*
|
|
||||||
* info_req request type definitions for meminfo
|
|
||||||
* request types starting with MEMINFO_V are used for Virtual addresses
|
|
||||||
* and should not be mixed with MEMINFO_PLGRP which is targeted for Physical
|
|
||||||
* addresses
|
|
||||||
*/
|
|
||||||
# define MEMINFO_SHIFT 16
|
|
||||||
# define MEMINFO_MASK (0xFF << MEMINFO_SHIFT)
|
|
||||||
# define MEMINFO_VPHYSICAL (0x01 << MEMINFO_SHIFT) /* get physical addr */
|
|
||||||
# define MEMINFO_VLGRP (0x02 << MEMINFO_SHIFT) /* get lgroup */
|
|
||||||
# define MEMINFO_VPAGESIZE (0x03 << MEMINFO_SHIFT) /* size of phys page */
|
|
||||||
# define MEMINFO_VREPLCNT (0x04 << MEMINFO_SHIFT) /* no. of replica */
|
|
||||||
# define MEMINFO_VREPL (0x05 << MEMINFO_SHIFT) /* physical replica */
|
|
||||||
# define MEMINFO_VREPL_LGRP (0x06 << MEMINFO_SHIFT) /* lgrp of replica */
|
|
||||||
# define MEMINFO_PLGRP (0x07 << MEMINFO_SHIFT) /* lgroup for paddr */
|
|
||||||
|
|
||||||
/* maximum number of addresses meminfo() can process at a time */
|
|
||||||
# define MAX_MEMINFO_CNT 256
|
|
||||||
|
|
||||||
/* maximum number of request types */
|
|
||||||
# define MAX_MEMINFO_REQ 31
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// see thr_setprio(3T) for the basis of these numbers
|
// see thr_setprio(3T) for the basis of these numbers
|
||||||
#define MinimumPriority 0
|
#define MinimumPriority 0
|
||||||
#define NormalPriority 64
|
#define NormalPriority 64
|
||||||
@ -1924,12 +1859,13 @@ bool os::address_is_in_vm(address addr) {
|
|||||||
Dl_info dlinfo;
|
Dl_info dlinfo;
|
||||||
|
|
||||||
if (libjvm_base_addr == NULL) {
|
if (libjvm_base_addr == NULL) {
|
||||||
dladdr(CAST_FROM_FN_PTR(void *, os::address_is_in_vm), &dlinfo);
|
if (dladdr(CAST_FROM_FN_PTR(void *, os::address_is_in_vm), &dlinfo) != 0) {
|
||||||
libjvm_base_addr = (address)dlinfo.dli_fbase;
|
libjvm_base_addr = (address)dlinfo.dli_fbase;
|
||||||
|
}
|
||||||
assert(libjvm_base_addr !=NULL, "Cannot obtain base address for libjvm");
|
assert(libjvm_base_addr !=NULL, "Cannot obtain base address for libjvm");
|
||||||
}
|
}
|
||||||
|
|
||||||
if (dladdr((void *)addr, &dlinfo)) {
|
if (dladdr((void *)addr, &dlinfo) != 0) {
|
||||||
if (libjvm_base_addr == (address)dlinfo.dli_fbase) return true;
|
if (libjvm_base_addr == (address)dlinfo.dli_fbase) return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1941,10 +1877,13 @@ static dladdr1_func_type dladdr1_func = NULL;
|
|||||||
|
|
||||||
bool os::dll_address_to_function_name(address addr, char *buf,
|
bool os::dll_address_to_function_name(address addr, char *buf,
|
||||||
int buflen, int * offset) {
|
int buflen, int * offset) {
|
||||||
|
// buf is not optional, but offset is optional
|
||||||
|
assert(buf != NULL, "sanity check");
|
||||||
|
|
||||||
Dl_info dlinfo;
|
Dl_info dlinfo;
|
||||||
|
|
||||||
// dladdr1_func was initialized in os::init()
|
// dladdr1_func was initialized in os::init()
|
||||||
if (dladdr1_func){
|
if (dladdr1_func != NULL) {
|
||||||
// yes, we have dladdr1
|
// yes, we have dladdr1
|
||||||
|
|
||||||
// Support for dladdr1 is checked at runtime; it may be
|
// Support for dladdr1 is checked at runtime; it may be
|
||||||
@ -1960,59 +1899,74 @@ bool os::dll_address_to_function_name(address addr, char *buf,
|
|||||||
Elf32_Sym * info;
|
Elf32_Sym * info;
|
||||||
#endif
|
#endif
|
||||||
if (dladdr1_func((void *)addr, &dlinfo, (void **)&info,
|
if (dladdr1_func((void *)addr, &dlinfo, (void **)&info,
|
||||||
RTLD_DL_SYMENT)) {
|
RTLD_DL_SYMENT) != 0) {
|
||||||
if ((char *)dlinfo.dli_saddr + info->st_size > (char *)addr) {
|
// see if we have a matching symbol that covers our address
|
||||||
if (buf != NULL) {
|
if (dlinfo.dli_saddr != NULL &&
|
||||||
if (!Decoder::demangle(dlinfo.dli_sname, buf, buflen))
|
(char *)dlinfo.dli_saddr + info->st_size > (char *)addr) {
|
||||||
|
if (dlinfo.dli_sname != NULL) {
|
||||||
|
if (!Decoder::demangle(dlinfo.dli_sname, buf, buflen)) {
|
||||||
jio_snprintf(buf, buflen, "%s", dlinfo.dli_sname);
|
jio_snprintf(buf, buflen, "%s", dlinfo.dli_sname);
|
||||||
}
|
}
|
||||||
if (offset != NULL) *offset = addr - (address)dlinfo.dli_saddr;
|
if (offset != NULL) *offset = addr - (address)dlinfo.dli_saddr;
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (dlinfo.dli_fname != NULL && dlinfo.dli_fbase != 0) {
|
// no matching symbol so try for just file info
|
||||||
|
if (dlinfo.dli_fname != NULL && dlinfo.dli_fbase != NULL) {
|
||||||
if (Decoder::decode((address)(addr - (address)dlinfo.dli_fbase),
|
if (Decoder::decode((address)(addr - (address)dlinfo.dli_fbase),
|
||||||
buf, buflen, offset, dlinfo.dli_fname)) {
|
buf, buflen, offset, dlinfo.dli_fname)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (buf != NULL) buf[0] = '\0';
|
}
|
||||||
|
buf[0] = '\0';
|
||||||
if (offset != NULL) *offset = -1;
|
if (offset != NULL) *offset = -1;
|
||||||
return false;
|
return false;
|
||||||
} else {
|
}
|
||||||
|
|
||||||
// no, only dladdr is available
|
// no, only dladdr is available
|
||||||
if (dladdr((void *)addr, &dlinfo)) {
|
if (dladdr((void *)addr, &dlinfo) != 0) {
|
||||||
if (buf != NULL) {
|
// see if we have a matching symbol
|
||||||
if (!Decoder::demangle(dlinfo.dli_sname, buf, buflen))
|
if (dlinfo.dli_saddr != NULL && dlinfo.dli_sname != NULL) {
|
||||||
|
if (!Decoder::demangle(dlinfo.dli_sname, buf, buflen)) {
|
||||||
jio_snprintf(buf, buflen, dlinfo.dli_sname);
|
jio_snprintf(buf, buflen, dlinfo.dli_sname);
|
||||||
}
|
}
|
||||||
if (offset != NULL) *offset = addr - (address)dlinfo.dli_saddr;
|
if (offset != NULL) *offset = addr - (address)dlinfo.dli_saddr;
|
||||||
return true;
|
return true;
|
||||||
} else if (dlinfo.dli_fname != NULL && dlinfo.dli_fbase != 0) {
|
}
|
||||||
|
// no matching symbol so try for just file info
|
||||||
|
if (dlinfo.dli_fname != NULL && dlinfo.dli_fbase != NULL) {
|
||||||
if (Decoder::decode((address)(addr - (address)dlinfo.dli_fbase),
|
if (Decoder::decode((address)(addr - (address)dlinfo.dli_fbase),
|
||||||
buf, buflen, offset, dlinfo.dli_fname)) {
|
buf, buflen, offset, dlinfo.dli_fname)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (buf != NULL) buf[0] = '\0';
|
}
|
||||||
|
buf[0] = '\0';
|
||||||
if (offset != NULL) *offset = -1;
|
if (offset != NULL) *offset = -1;
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool os::dll_address_to_library_name(address addr, char* buf,
|
bool os::dll_address_to_library_name(address addr, char* buf,
|
||||||
int buflen, int* offset) {
|
int buflen, int* offset) {
|
||||||
|
// buf is not optional, but offset is optional
|
||||||
|
assert(buf != NULL, "sanity check");
|
||||||
|
|
||||||
Dl_info dlinfo;
|
Dl_info dlinfo;
|
||||||
|
|
||||||
if (dladdr((void*)addr, &dlinfo)){
|
if (dladdr((void*)addr, &dlinfo) != 0) {
|
||||||
if (buf) jio_snprintf(buf, buflen, "%s", dlinfo.dli_fname);
|
if (dlinfo.dli_fname != NULL) {
|
||||||
if (offset) *offset = addr - (address)dlinfo.dli_fbase;
|
jio_snprintf(buf, buflen, "%s", dlinfo.dli_fname);
|
||||||
|
}
|
||||||
|
if (dlinfo.dli_fbase != NULL && offset != NULL) {
|
||||||
|
*offset = addr - (address)dlinfo.dli_fbase;
|
||||||
|
}
|
||||||
return true;
|
return true;
|
||||||
} else {
|
}
|
||||||
if (buf) buf[0] = '\0';
|
|
||||||
|
buf[0] = '\0';
|
||||||
if (offset) *offset = -1;
|
if (offset) *offset = -1;
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Prints the names and full paths of all opened dynamic libraries
|
// Prints the names and full paths of all opened dynamic libraries
|
||||||
@ -2025,7 +1979,8 @@ void os::print_dll_info(outputStream * st) {
|
|||||||
|
|
||||||
st->print_cr("Dynamic libraries:"); st->flush();
|
st->print_cr("Dynamic libraries:"); st->flush();
|
||||||
|
|
||||||
if (!dladdr(CAST_FROM_FN_PTR(void *, os::print_dll_info), &dli)) {
|
if (dladdr(CAST_FROM_FN_PTR(void *, os::print_dll_info), &dli) == 0 ||
|
||||||
|
dli.dli_fname == NULL) {
|
||||||
st->print_cr("Error: Cannot print dynamic libraries.");
|
st->print_cr("Error: Cannot print dynamic libraries.");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -2475,7 +2430,12 @@ void os::jvm_path(char *buf, jint buflen) {
|
|||||||
Dl_info dlinfo;
|
Dl_info dlinfo;
|
||||||
int ret = dladdr(CAST_FROM_FN_PTR(void *, os::jvm_path), &dlinfo);
|
int ret = dladdr(CAST_FROM_FN_PTR(void *, os::jvm_path), &dlinfo);
|
||||||
assert(ret != 0, "cannot locate libjvm");
|
assert(ret != 0, "cannot locate libjvm");
|
||||||
|
if (ret != 0 && dlinfo.dli_fname != NULL) {
|
||||||
realpath((char *)dlinfo.dli_fname, buf);
|
realpath((char *)dlinfo.dli_fname, buf);
|
||||||
|
} else {
|
||||||
|
buf[0] = '\0';
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
if (Arguments::created_by_gamma_launcher()) {
|
if (Arguments::created_by_gamma_launcher()) {
|
||||||
// Support for the gamma launcher. Typical value for buf is
|
// Support for the gamma launcher. Typical value for buf is
|
||||||
@ -2859,7 +2819,7 @@ int os::Solaris::commit_memory_impl(char* addr, size_t bytes,
|
|||||||
size_t alignment_hint, bool exec) {
|
size_t alignment_hint, bool exec) {
|
||||||
int err = Solaris::commit_memory_impl(addr, bytes, exec);
|
int err = Solaris::commit_memory_impl(addr, bytes, exec);
|
||||||
if (err == 0) {
|
if (err == 0) {
|
||||||
if (UseMPSS && alignment_hint > (size_t)vm_page_size()) {
|
if (UseLargePages && (alignment_hint > (size_t)vm_page_size())) {
|
||||||
// If the large page size has been set and the VM
|
// If the large page size has been set and the VM
|
||||||
// is using large pages, use the large page size
|
// is using large pages, use the large page size
|
||||||
// if it is smaller than the alignment hint. This is
|
// if it is smaller than the alignment hint. This is
|
||||||
@ -2878,7 +2838,7 @@ int os::Solaris::commit_memory_impl(char* addr, size_t bytes,
|
|||||||
page_size = alignment_hint;
|
page_size = alignment_hint;
|
||||||
}
|
}
|
||||||
// Since this is a hint, ignore any failures.
|
// Since this is a hint, ignore any failures.
|
||||||
(void)Solaris::set_mpss_range(addr, bytes, page_size);
|
(void)Solaris::setup_large_pages(addr, bytes, page_size);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return err;
|
return err;
|
||||||
@ -2921,8 +2881,8 @@ bool os::remove_stack_guard_pages(char* addr, size_t size) {
|
|||||||
void os::pd_realign_memory(char *addr, size_t bytes, size_t alignment_hint) {
|
void os::pd_realign_memory(char *addr, size_t bytes, size_t alignment_hint) {
|
||||||
assert((intptr_t)addr % alignment_hint == 0, "Address should be aligned.");
|
assert((intptr_t)addr % alignment_hint == 0, "Address should be aligned.");
|
||||||
assert((intptr_t)(addr + bytes) % alignment_hint == 0, "End should be aligned.");
|
assert((intptr_t)(addr + bytes) % alignment_hint == 0, "End should be aligned.");
|
||||||
if (UseLargePages && UseMPSS) {
|
if (UseLargePages) {
|
||||||
Solaris::set_mpss_range(addr, bytes, alignment_hint);
|
Solaris::setup_large_pages(addr, bytes, alignment_hint);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -3321,47 +3281,8 @@ bool os::unguard_memory(char* addr, size_t bytes) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Large page support
|
// Large page support
|
||||||
|
|
||||||
// UseLargePages is the master flag to enable/disable large page memory.
|
|
||||||
// UseMPSS and UseISM are supported for compatibility reasons. Their combined
|
|
||||||
// effects can be described in the following table:
|
|
||||||
//
|
|
||||||
// UseLargePages UseMPSS UseISM
|
|
||||||
// false * * => UseLargePages is the master switch, turning
|
|
||||||
// it off will turn off both UseMPSS and
|
|
||||||
// UseISM. VM will not use large page memory
|
|
||||||
// regardless the settings of UseMPSS/UseISM.
|
|
||||||
// true false false => Unless future Solaris provides other
|
|
||||||
// mechanism to use large page memory, this
|
|
||||||
// combination is equivalent to -UseLargePages,
|
|
||||||
// VM will not use large page memory
|
|
||||||
// true true false => JVM will use MPSS for large page memory.
|
|
||||||
// This is the default behavior.
|
|
||||||
// true false true => JVM will use ISM for large page memory.
|
|
||||||
// true true true => JVM will use ISM if it is available.
|
|
||||||
// Otherwise, JVM will fall back to MPSS.
|
|
||||||
// Becaues ISM is now available on all
|
|
||||||
// supported Solaris versions, this combination
|
|
||||||
// is equivalent to +UseISM -UseMPSS.
|
|
||||||
|
|
||||||
static size_t _large_page_size = 0;
|
static size_t _large_page_size = 0;
|
||||||
|
|
||||||
bool os::Solaris::ism_sanity_check(bool warn, size_t * page_size) {
|
|
||||||
// x86 uses either 2M or 4M page, depending on whether PAE (Physical Address
|
|
||||||
// Extensions) mode is enabled. AMD64/EM64T uses 2M page in 64bit mode. Sparc
|
|
||||||
// can support multiple page sizes.
|
|
||||||
|
|
||||||
// Don't bother to probe page size because getpagesizes() comes with MPSS.
|
|
||||||
// ISM is only recommended on old Solaris where there is no MPSS support.
|
|
||||||
// Simply choose a conservative value as default.
|
|
||||||
*page_size = LargePageSizeInBytes ? LargePageSizeInBytes :
|
|
||||||
SPARC_ONLY(4 * M) IA32_ONLY(4 * M) AMD64_ONLY(2 * M)
|
|
||||||
ARM_ONLY(2 * M);
|
|
||||||
|
|
||||||
// ISM is available on all supported Solaris versions
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Insertion sort for small arrays (descending order).
|
// Insertion sort for small arrays (descending order).
|
||||||
static void insertion_sort_descending(size_t* array, int len) {
|
static void insertion_sort_descending(size_t* array, int len) {
|
||||||
for (int i = 0; i < len; i++) {
|
for (int i = 0; i < len; i++) {
|
||||||
@ -3374,7 +3295,7 @@ static void insertion_sort_descending(size_t* array, int len) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool os::Solaris::mpss_sanity_check(bool warn, size_t * page_size) {
|
bool os::Solaris::mpss_sanity_check(bool warn, size_t* page_size) {
|
||||||
const unsigned int usable_count = VM_Version::page_size_count();
|
const unsigned int usable_count = VM_Version::page_size_count();
|
||||||
if (usable_count == 1) {
|
if (usable_count == 1) {
|
||||||
return false;
|
return false;
|
||||||
@ -3440,41 +3361,24 @@ bool os::Solaris::mpss_sanity_check(bool warn, size_t * page_size) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void os::large_page_init() {
|
void os::large_page_init() {
|
||||||
if (!UseLargePages) {
|
if (UseLargePages) {
|
||||||
UseISM = false;
|
|
||||||
UseMPSS = false;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// print a warning if any large page related flag is specified on command line
|
// print a warning if any large page related flag is specified on command line
|
||||||
bool warn_on_failure = !FLAG_IS_DEFAULT(UseLargePages) ||
|
bool warn_on_failure = !FLAG_IS_DEFAULT(UseLargePages) ||
|
||||||
!FLAG_IS_DEFAULT(UseISM) ||
|
|
||||||
!FLAG_IS_DEFAULT(UseMPSS) ||
|
|
||||||
!FLAG_IS_DEFAULT(LargePageSizeInBytes);
|
!FLAG_IS_DEFAULT(LargePageSizeInBytes);
|
||||||
UseISM = UseISM &&
|
|
||||||
Solaris::ism_sanity_check(warn_on_failure, &_large_page_size);
|
UseLargePages = Solaris::mpss_sanity_check(warn_on_failure, &_large_page_size);
|
||||||
if (UseISM) {
|
|
||||||
// ISM disables MPSS to be compatible with old JDK behavior
|
|
||||||
UseMPSS = false;
|
|
||||||
_page_sizes[0] = _large_page_size;
|
|
||||||
_page_sizes[1] = vm_page_size();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
UseMPSS = UseMPSS &&
|
|
||||||
Solaris::mpss_sanity_check(warn_on_failure, &_large_page_size);
|
|
||||||
|
|
||||||
UseLargePages = UseISM || UseMPSS;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool os::Solaris::set_mpss_range(caddr_t start, size_t bytes, size_t align) {
|
bool os::Solaris::setup_large_pages(caddr_t start, size_t bytes, size_t align) {
|
||||||
// Signal to OS that we want large pages for addresses
|
// Signal to OS that we want large pages for addresses
|
||||||
// from addr, addr + bytes
|
// from addr, addr + bytes
|
||||||
struct memcntl_mha mpss_struct;
|
struct memcntl_mha mpss_struct;
|
||||||
mpss_struct.mha_cmd = MHA_MAPSIZE_VA;
|
mpss_struct.mha_cmd = MHA_MAPSIZE_VA;
|
||||||
mpss_struct.mha_pagesize = align;
|
mpss_struct.mha_pagesize = align;
|
||||||
mpss_struct.mha_flags = 0;
|
mpss_struct.mha_flags = 0;
|
||||||
if (memcntl(start, bytes, MC_HAT_ADVISE,
|
// Upon successful completion, memcntl() returns 0
|
||||||
(caddr_t) &mpss_struct, 0, 0) < 0) {
|
if (memcntl(start, bytes, MC_HAT_ADVISE, (caddr_t) &mpss_struct, 0, 0)) {
|
||||||
debug_only(warning("Attempt to use MPSS failed."));
|
debug_only(warning("Attempt to use MPSS failed."));
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
@ -3482,72 +3386,13 @@ bool os::Solaris::set_mpss_range(caddr_t start, size_t bytes, size_t align) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
char* os::reserve_memory_special(size_t size, char* addr, bool exec) {
|
char* os::reserve_memory_special(size_t size, char* addr, bool exec) {
|
||||||
// "exec" is passed in but not used. Creating the shared image for
|
fatal("os::reserve_memory_special should not be called on Solaris.");
|
||||||
// the code cache doesn't have an SHM_X executable permission to check.
|
|
||||||
assert(UseLargePages && UseISM, "only for ISM large pages");
|
|
||||||
|
|
||||||
char* retAddr = NULL;
|
|
||||||
int shmid;
|
|
||||||
key_t ismKey;
|
|
||||||
|
|
||||||
bool warn_on_failure = UseISM &&
|
|
||||||
(!FLAG_IS_DEFAULT(UseLargePages) ||
|
|
||||||
!FLAG_IS_DEFAULT(UseISM) ||
|
|
||||||
!FLAG_IS_DEFAULT(LargePageSizeInBytes)
|
|
||||||
);
|
|
||||||
char msg[128];
|
|
||||||
|
|
||||||
ismKey = IPC_PRIVATE;
|
|
||||||
|
|
||||||
// Create a large shared memory region to attach to based on size.
|
|
||||||
// Currently, size is the total size of the heap
|
|
||||||
shmid = shmget(ismKey, size, SHM_R | SHM_W | IPC_CREAT);
|
|
||||||
if (shmid == -1){
|
|
||||||
if (warn_on_failure) {
|
|
||||||
jio_snprintf(msg, sizeof(msg), "Failed to reserve shared memory (errno = %d).", errno);
|
|
||||||
warning(msg);
|
|
||||||
}
|
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
|
||||||
|
|
||||||
// Attach to the region
|
|
||||||
retAddr = (char *) shmat(shmid, 0, SHM_SHARE_MMU | SHM_R | SHM_W);
|
|
||||||
int err = errno;
|
|
||||||
|
|
||||||
// Remove shmid. If shmat() is successful, the actual shared memory segment
|
|
||||||
// will be deleted when it's detached by shmdt() or when the process
|
|
||||||
// terminates. If shmat() is not successful this will remove the shared
|
|
||||||
// segment immediately.
|
|
||||||
shmctl(shmid, IPC_RMID, NULL);
|
|
||||||
|
|
||||||
if (retAddr == (char *) -1) {
|
|
||||||
if (warn_on_failure) {
|
|
||||||
jio_snprintf(msg, sizeof(msg), "Failed to attach shared memory (errno = %d).", err);
|
|
||||||
warning(msg);
|
|
||||||
}
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
if ((retAddr != NULL) && UseNUMAInterleaving) {
|
|
||||||
numa_make_global(retAddr, size);
|
|
||||||
}
|
|
||||||
|
|
||||||
// The memory is committed
|
|
||||||
MemTracker::record_virtual_memory_reserve_and_commit((address)retAddr, size, mtNone, CURRENT_PC);
|
|
||||||
|
|
||||||
return retAddr;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool os::release_memory_special(char* base, size_t bytes) {
|
bool os::release_memory_special(char* base, size_t bytes) {
|
||||||
MemTracker::Tracker tkr = MemTracker::get_virtual_memory_release_tracker();
|
fatal("os::release_memory_special should not be called on Solaris.");
|
||||||
// detaching the SHM segment will also delete it, see reserve_memory_special()
|
|
||||||
int rslt = shmdt(base);
|
|
||||||
if (rslt == 0) {
|
|
||||||
tkr.record((address)base, bytes);
|
|
||||||
return true;
|
|
||||||
} else {
|
|
||||||
tkr.discard();
|
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
size_t os::large_page_size() {
|
size_t os::large_page_size() {
|
||||||
@ -3557,11 +3402,11 @@ size_t os::large_page_size() {
|
|||||||
// MPSS allows application to commit large page memory on demand; with ISM
|
// MPSS allows application to commit large page memory on demand; with ISM
|
||||||
// the entire memory region must be allocated as shared memory.
|
// the entire memory region must be allocated as shared memory.
|
||||||
bool os::can_commit_large_page_memory() {
|
bool os::can_commit_large_page_memory() {
|
||||||
return UseISM ? false : true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
bool os::can_execute_large_page_memory() {
|
bool os::can_execute_large_page_memory() {
|
||||||
return UseISM ? false : true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
static int os_sleep(jlong millis, bool interruptible) {
|
static int os_sleep(jlong millis, bool interruptible) {
|
||||||
@ -3835,28 +3680,6 @@ static bool priocntl_enable = false;
|
|||||||
static const int criticalPrio = 60; // FX/60 is critical thread class/priority on T4
|
static const int criticalPrio = 60; // FX/60 is critical thread class/priority on T4
|
||||||
static int java_MaxPriority_to_os_priority = 0; // Saved mapping
|
static int java_MaxPriority_to_os_priority = 0; // Saved mapping
|
||||||
|
|
||||||
// Call the version of priocntl suitable for all supported versions
|
|
||||||
// of Solaris. We need to call through this wrapper so that we can
|
|
||||||
// build on Solaris 9 and run on Solaris 8, 9 and 10.
|
|
||||||
//
|
|
||||||
// This code should be removed if we ever stop supporting Solaris 8
|
|
||||||
// and earlier releases.
|
|
||||||
|
|
||||||
static long priocntl_stub(int pcver, idtype_t idtype, id_t id, int cmd, caddr_t arg);
|
|
||||||
typedef long (*priocntl_type)(int pcver, idtype_t idtype, id_t id, int cmd, caddr_t arg);
|
|
||||||
static priocntl_type priocntl_ptr = priocntl_stub;
|
|
||||||
|
|
||||||
// Stub to set the value of the real pointer, and then call the real
|
|
||||||
// function.
|
|
||||||
|
|
||||||
static long priocntl_stub(int pcver, idtype_t idtype, id_t id, int cmd, caddr_t arg) {
|
|
||||||
// Try Solaris 8- name only.
|
|
||||||
priocntl_type tmp = (priocntl_type)dlsym(RTLD_DEFAULT, "__priocntl");
|
|
||||||
guarantee(tmp != NULL, "priocntl function not found.");
|
|
||||||
priocntl_ptr = tmp;
|
|
||||||
return (*priocntl_ptr)(PC_VERSION, idtype, id, cmd, arg);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// lwp_priocntl_init
|
// lwp_priocntl_init
|
||||||
//
|
//
|
||||||
@ -3864,9 +3687,7 @@ static long priocntl_stub(int pcver, idtype_t idtype, id_t id, int cmd, caddr_t
|
|||||||
//
|
//
|
||||||
// Return errno or 0 if OK.
|
// Return errno or 0 if OK.
|
||||||
//
|
//
|
||||||
static
|
static int lwp_priocntl_init () {
|
||||||
int lwp_priocntl_init ()
|
|
||||||
{
|
|
||||||
int rslt;
|
int rslt;
|
||||||
pcinfo_t ClassInfo;
|
pcinfo_t ClassInfo;
|
||||||
pcparms_t ParmInfo;
|
pcparms_t ParmInfo;
|
||||||
@ -3906,7 +3727,7 @@ int lwp_priocntl_init ()
|
|||||||
|
|
||||||
strcpy(ClassInfo.pc_clname, "TS");
|
strcpy(ClassInfo.pc_clname, "TS");
|
||||||
ClassInfo.pc_cid = -1;
|
ClassInfo.pc_cid = -1;
|
||||||
rslt = (*priocntl_ptr)(PC_VERSION, P_ALL, 0, PC_GETCID, (caddr_t)&ClassInfo);
|
rslt = priocntl(P_ALL, 0, PC_GETCID, (caddr_t)&ClassInfo);
|
||||||
if (rslt < 0) return errno;
|
if (rslt < 0) return errno;
|
||||||
assert(ClassInfo.pc_cid != -1, "cid for TS class is -1");
|
assert(ClassInfo.pc_cid != -1, "cid for TS class is -1");
|
||||||
tsLimits.schedPolicy = ClassInfo.pc_cid;
|
tsLimits.schedPolicy = ClassInfo.pc_cid;
|
||||||
@ -3915,7 +3736,7 @@ int lwp_priocntl_init ()
|
|||||||
|
|
||||||
strcpy(ClassInfo.pc_clname, "IA");
|
strcpy(ClassInfo.pc_clname, "IA");
|
||||||
ClassInfo.pc_cid = -1;
|
ClassInfo.pc_cid = -1;
|
||||||
rslt = (*priocntl_ptr)(PC_VERSION, P_ALL, 0, PC_GETCID, (caddr_t)&ClassInfo);
|
rslt = priocntl(P_ALL, 0, PC_GETCID, (caddr_t)&ClassInfo);
|
||||||
if (rslt < 0) return errno;
|
if (rslt < 0) return errno;
|
||||||
assert(ClassInfo.pc_cid != -1, "cid for IA class is -1");
|
assert(ClassInfo.pc_cid != -1, "cid for IA class is -1");
|
||||||
iaLimits.schedPolicy = ClassInfo.pc_cid;
|
iaLimits.schedPolicy = ClassInfo.pc_cid;
|
||||||
@ -3924,7 +3745,7 @@ int lwp_priocntl_init ()
|
|||||||
|
|
||||||
strcpy(ClassInfo.pc_clname, "RT");
|
strcpy(ClassInfo.pc_clname, "RT");
|
||||||
ClassInfo.pc_cid = -1;
|
ClassInfo.pc_cid = -1;
|
||||||
rslt = (*priocntl_ptr)(PC_VERSION, P_ALL, 0, PC_GETCID, (caddr_t)&ClassInfo);
|
rslt = priocntl(P_ALL, 0, PC_GETCID, (caddr_t)&ClassInfo);
|
||||||
if (rslt < 0) return errno;
|
if (rslt < 0) return errno;
|
||||||
assert(ClassInfo.pc_cid != -1, "cid for RT class is -1");
|
assert(ClassInfo.pc_cid != -1, "cid for RT class is -1");
|
||||||
rtLimits.schedPolicy = ClassInfo.pc_cid;
|
rtLimits.schedPolicy = ClassInfo.pc_cid;
|
||||||
@ -3933,7 +3754,7 @@ int lwp_priocntl_init ()
|
|||||||
|
|
||||||
strcpy(ClassInfo.pc_clname, "FX");
|
strcpy(ClassInfo.pc_clname, "FX");
|
||||||
ClassInfo.pc_cid = -1;
|
ClassInfo.pc_cid = -1;
|
||||||
rslt = (*priocntl_ptr)(PC_VERSION, P_ALL, 0, PC_GETCID, (caddr_t)&ClassInfo);
|
rslt = priocntl(P_ALL, 0, PC_GETCID, (caddr_t)&ClassInfo);
|
||||||
if (rslt < 0) return errno;
|
if (rslt < 0) return errno;
|
||||||
assert(ClassInfo.pc_cid != -1, "cid for FX class is -1");
|
assert(ClassInfo.pc_cid != -1, "cid for FX class is -1");
|
||||||
fxLimits.schedPolicy = ClassInfo.pc_cid;
|
fxLimits.schedPolicy = ClassInfo.pc_cid;
|
||||||
@ -3944,7 +3765,7 @@ int lwp_priocntl_init ()
|
|||||||
// This will normally be IA, TS or, rarely, FX or RT.
|
// This will normally be IA, TS or, rarely, FX or RT.
|
||||||
memset(&ParmInfo, 0, sizeof(ParmInfo));
|
memset(&ParmInfo, 0, sizeof(ParmInfo));
|
||||||
ParmInfo.pc_cid = PC_CLNULL;
|
ParmInfo.pc_cid = PC_CLNULL;
|
||||||
rslt = (*priocntl_ptr) (PC_VERSION, P_PID, P_MYID, PC_GETPARMS, (caddr_t)&ParmInfo);
|
rslt = priocntl(P_PID, P_MYID, PC_GETPARMS, (caddr_t)&ParmInfo);
|
||||||
if (rslt < 0) return errno;
|
if (rslt < 0) return errno;
|
||||||
myClass = ParmInfo.pc_cid;
|
myClass = ParmInfo.pc_cid;
|
||||||
|
|
||||||
@ -3952,7 +3773,7 @@ int lwp_priocntl_init ()
|
|||||||
// about the class.
|
// about the class.
|
||||||
ClassInfo.pc_cid = myClass;
|
ClassInfo.pc_cid = myClass;
|
||||||
ClassInfo.pc_clname[0] = 0;
|
ClassInfo.pc_clname[0] = 0;
|
||||||
rslt = (*priocntl_ptr) (PC_VERSION, (idtype)0, 0, PC_GETCLINFO, (caddr_t)&ClassInfo);
|
rslt = priocntl((idtype)0, 0, PC_GETCLINFO, (caddr_t)&ClassInfo);
|
||||||
if (rslt < 0) return errno;
|
if (rslt < 0) return errno;
|
||||||
|
|
||||||
if (ThreadPriorityVerbose) {
|
if (ThreadPriorityVerbose) {
|
||||||
@ -3961,7 +3782,7 @@ int lwp_priocntl_init ()
|
|||||||
|
|
||||||
memset(&ParmInfo, 0, sizeof(pcparms_t));
|
memset(&ParmInfo, 0, sizeof(pcparms_t));
|
||||||
ParmInfo.pc_cid = PC_CLNULL;
|
ParmInfo.pc_cid = PC_CLNULL;
|
||||||
rslt = (*priocntl_ptr)(PC_VERSION, P_PID, P_MYID, PC_GETPARMS, (caddr_t)&ParmInfo);
|
rslt = priocntl(P_PID, P_MYID, PC_GETPARMS, (caddr_t)&ParmInfo);
|
||||||
if (rslt < 0) return errno;
|
if (rslt < 0) return errno;
|
||||||
|
|
||||||
if (ParmInfo.pc_cid == rtLimits.schedPolicy) {
|
if (ParmInfo.pc_cid == rtLimits.schedPolicy) {
|
||||||
@ -4065,7 +3886,7 @@ int set_lwp_class_and_priority(int ThreadID, int lwpid,
|
|||||||
|
|
||||||
memset(&ParmInfo, 0, sizeof(pcparms_t));
|
memset(&ParmInfo, 0, sizeof(pcparms_t));
|
||||||
ParmInfo.pc_cid = PC_CLNULL;
|
ParmInfo.pc_cid = PC_CLNULL;
|
||||||
rslt = (*priocntl_ptr)(PC_VERSION, P_LWPID, lwpid, PC_GETPARMS, (caddr_t)&ParmInfo);
|
rslt = priocntl(P_LWPID, lwpid, PC_GETPARMS, (caddr_t)&ParmInfo);
|
||||||
if (rslt < 0) return errno;
|
if (rslt < 0) return errno;
|
||||||
|
|
||||||
int cur_class = ParmInfo.pc_cid;
|
int cur_class = ParmInfo.pc_cid;
|
||||||
@ -4133,7 +3954,7 @@ int set_lwp_class_and_priority(int ThreadID, int lwpid,
|
|||||||
return EINVAL; // no clue, punt
|
return EINVAL; // no clue, punt
|
||||||
}
|
}
|
||||||
|
|
||||||
rslt = (*priocntl_ptr)(PC_VERSION, P_LWPID, lwpid, PC_SETPARMS, (caddr_t)&ParmInfo);
|
rslt = priocntl(P_LWPID, lwpid, PC_SETPARMS, (caddr_t)&ParmInfo);
|
||||||
if (ThreadPriorityVerbose && rslt) {
|
if (ThreadPriorityVerbose && rslt) {
|
||||||
tty->print_cr ("PC_SETPARMS ->%d %d\n", rslt, errno);
|
tty->print_cr ("PC_SETPARMS ->%d %d\n", rslt, errno);
|
||||||
}
|
}
|
||||||
@ -4152,7 +3973,7 @@ int set_lwp_class_and_priority(int ThreadID, int lwpid,
|
|||||||
|
|
||||||
memset(&ReadBack, 0, sizeof(pcparms_t));
|
memset(&ReadBack, 0, sizeof(pcparms_t));
|
||||||
ReadBack.pc_cid = PC_CLNULL;
|
ReadBack.pc_cid = PC_CLNULL;
|
||||||
rslt = (*priocntl_ptr)(PC_VERSION, P_LWPID, lwpid, PC_GETPARMS, (caddr_t)&ReadBack);
|
rslt = priocntl(P_LWPID, lwpid, PC_GETPARMS, (caddr_t)&ReadBack);
|
||||||
assert(rslt >= 0, "priocntl failed");
|
assert(rslt >= 0, "priocntl failed");
|
||||||
Actual = Expected = 0xBAD;
|
Actual = Expected = 0xBAD;
|
||||||
assert(ParmInfo.pc_cid == ReadBack.pc_cid, "cid's don't match");
|
assert(ParmInfo.pc_cid == ReadBack.pc_cid, "cid's don't match");
|
||||||
@ -5244,11 +5065,6 @@ uint_t os::Solaris::getisax(uint32_t* array, uint_t n) {
|
|||||||
return _getisax(array, n);
|
return _getisax(array, n);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Symbol doesn't exist in Solaris 8 pset.h
|
|
||||||
#ifndef PS_MYID
|
|
||||||
#define PS_MYID -3
|
|
||||||
#endif
|
|
||||||
|
|
||||||
// int pset_getloadavg(psetid_t pset, double loadavg[], int nelem);
|
// int pset_getloadavg(psetid_t pset, double loadavg[], int nelem);
|
||||||
typedef long (*pset_getloadavg_type)(psetid_t pset, double loadavg[], int nelem);
|
typedef long (*pset_getloadavg_type)(psetid_t pset, double loadavg[], int nelem);
|
||||||
static pset_getloadavg_type pset_getloadavg_ptr = NULL;
|
static pset_getloadavg_type pset_getloadavg_ptr = NULL;
|
||||||
@ -5418,20 +5234,6 @@ jint os::init_2(void) {
|
|||||||
UseNUMA = false;
|
UseNUMA = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// ISM is not compatible with the NUMA allocator - it always allocates
|
|
||||||
// pages round-robin across the lgroups.
|
|
||||||
if (UseNUMA && UseLargePages && UseISM) {
|
|
||||||
if (!FLAG_IS_DEFAULT(UseNUMA)) {
|
|
||||||
if (FLAG_IS_DEFAULT(UseLargePages) && FLAG_IS_DEFAULT(UseISM)) {
|
|
||||||
UseLargePages = false;
|
|
||||||
} else {
|
|
||||||
warning("UseNUMA is not compatible with ISM large pages, disabling NUMA allocator");
|
|
||||||
UseNUMA = false;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
UseNUMA = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (!UseNUMA && ForceNUMA) {
|
if (!UseNUMA && ForceNUMA) {
|
||||||
UseNUMA = true;
|
UseNUMA = true;
|
||||||
}
|
}
|
||||||
@ -6077,24 +5879,20 @@ int os::loadavg(double loadavg[], int nelem) {
|
|||||||
bool os::find(address addr, outputStream* st) {
|
bool os::find(address addr, outputStream* st) {
|
||||||
Dl_info dlinfo;
|
Dl_info dlinfo;
|
||||||
memset(&dlinfo, 0, sizeof(dlinfo));
|
memset(&dlinfo, 0, sizeof(dlinfo));
|
||||||
if (dladdr(addr, &dlinfo)) {
|
if (dladdr(addr, &dlinfo) != 0) {
|
||||||
#ifdef _LP64
|
st->print(PTR_FORMAT ": ", addr);
|
||||||
st->print("0x%016lx: ", addr);
|
if (dlinfo.dli_sname != NULL && dlinfo.dli_saddr != NULL) {
|
||||||
#else
|
|
||||||
st->print("0x%08x: ", addr);
|
|
||||||
#endif
|
|
||||||
if (dlinfo.dli_sname != NULL)
|
|
||||||
st->print("%s+%#lx", dlinfo.dli_sname, addr-(intptr_t)dlinfo.dli_saddr);
|
st->print("%s+%#lx", dlinfo.dli_sname, addr-(intptr_t)dlinfo.dli_saddr);
|
||||||
else if (dlinfo.dli_fname)
|
} else if (dlinfo.dli_fbase != NULL)
|
||||||
st->print("<offset %#lx>", addr-(intptr_t)dlinfo.dli_fbase);
|
st->print("<offset %#lx>", addr-(intptr_t)dlinfo.dli_fbase);
|
||||||
else
|
else
|
||||||
st->print("<absolute address>");
|
st->print("<absolute address>");
|
||||||
if (dlinfo.dli_fname) st->print(" in %s", dlinfo.dli_fname);
|
if (dlinfo.dli_fname != NULL) {
|
||||||
#ifdef _LP64
|
st->print(" in %s", dlinfo.dli_fname);
|
||||||
if (dlinfo.dli_fbase) st->print(" at 0x%016lx", dlinfo.dli_fbase);
|
}
|
||||||
#else
|
if (dlinfo.dli_fbase != NULL) {
|
||||||
if (dlinfo.dli_fbase) st->print(" at 0x%08x", dlinfo.dli_fbase);
|
st->print(" at " PTR_FORMAT, dlinfo.dli_fbase);
|
||||||
#endif
|
}
|
||||||
st->cr();
|
st->cr();
|
||||||
|
|
||||||
if (Verbose) {
|
if (Verbose) {
|
||||||
@ -6105,7 +5903,7 @@ bool os::find(address addr, outputStream* st) {
|
|||||||
if (!lowest) lowest = (address) dlinfo.dli_fbase;
|
if (!lowest) lowest = (address) dlinfo.dli_fbase;
|
||||||
if (begin < lowest) begin = lowest;
|
if (begin < lowest) begin = lowest;
|
||||||
Dl_info dlinfo2;
|
Dl_info dlinfo2;
|
||||||
if (dladdr(end, &dlinfo2) && dlinfo2.dli_saddr != dlinfo.dli_saddr
|
if (dladdr(end, &dlinfo2) != 0 && dlinfo2.dli_saddr != dlinfo.dli_saddr
|
||||||
&& end > dlinfo2.dli_saddr && dlinfo2.dli_saddr > begin)
|
&& end > dlinfo2.dli_saddr && dlinfo2.dli_saddr > begin)
|
||||||
end = (address) dlinfo2.dli_saddr;
|
end = (address) dlinfo2.dli_saddr;
|
||||||
Disassembler::decode(begin, end, st);
|
Disassembler::decode(begin, end, st);
|
||||||
|
@ -106,8 +106,8 @@ class Solaris {
|
|||||||
|
|
||||||
static meminfo_func_t _meminfo;
|
static meminfo_func_t _meminfo;
|
||||||
|
|
||||||
// Large Page Support--mpss.
|
// Large Page Support
|
||||||
static bool set_mpss_range(caddr_t start, size_t bytes, size_t align);
|
static bool setup_large_pages(caddr_t start, size_t bytes, size_t align);
|
||||||
|
|
||||||
static void init_thread_fpu_state(void);
|
static void init_thread_fpu_state(void);
|
||||||
|
|
||||||
@ -174,7 +174,6 @@ class Solaris {
|
|||||||
static char* mmap_chunk(char *addr, size_t size, int flags, int prot);
|
static char* mmap_chunk(char *addr, size_t size, int flags, int prot);
|
||||||
static char* anon_mmap(char* requested_addr, size_t bytes, size_t alignment_hint, bool fixed);
|
static char* anon_mmap(char* requested_addr, size_t bytes, size_t alignment_hint, bool fixed);
|
||||||
static bool mpss_sanity_check(bool warn, size_t * page_size);
|
static bool mpss_sanity_check(bool warn, size_t * page_size);
|
||||||
static bool ism_sanity_check (bool warn, size_t * page_size);
|
|
||||||
|
|
||||||
// Workaround for 4352906. thr_stksegment sometimes returns
|
// Workaround for 4352906. thr_stksegment sometimes returns
|
||||||
// a bad value for the primordial thread's stack base when
|
// a bad value for the primordial thread's stack base when
|
||||||
|
@ -1420,6 +1420,9 @@ static int _locate_module_by_addr(int pid, char * mod_fname, address base_addr,
|
|||||||
|
|
||||||
bool os::dll_address_to_library_name(address addr, char* buf,
|
bool os::dll_address_to_library_name(address addr, char* buf,
|
||||||
int buflen, int* offset) {
|
int buflen, int* offset) {
|
||||||
|
// buf is not optional, but offset is optional
|
||||||
|
assert(buf != NULL, "sanity check");
|
||||||
|
|
||||||
// NOTE: the reason we don't use SymGetModuleInfo() is it doesn't always
|
// NOTE: the reason we don't use SymGetModuleInfo() is it doesn't always
|
||||||
// return the full path to the DLL file, sometimes it returns path
|
// return the full path to the DLL file, sometimes it returns path
|
||||||
// to the corresponding PDB file (debug info); sometimes it only
|
// to the corresponding PDB file (debug info); sometimes it only
|
||||||
@ -1434,20 +1437,23 @@ bool os::dll_address_to_library_name(address addr, char* buf,
|
|||||||
// buf already contains path name
|
// buf already contains path name
|
||||||
if (offset) *offset = addr - mi.base_addr;
|
if (offset) *offset = addr - mi.base_addr;
|
||||||
return true;
|
return true;
|
||||||
} else {
|
}
|
||||||
if (buf) buf[0] = '\0';
|
|
||||||
|
buf[0] = '\0';
|
||||||
if (offset) *offset = -1;
|
if (offset) *offset = -1;
|
||||||
return false;
|
return false;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
bool os::dll_address_to_function_name(address addr, char *buf,
|
bool os::dll_address_to_function_name(address addr, char *buf,
|
||||||
int buflen, int *offset) {
|
int buflen, int *offset) {
|
||||||
|
// buf is not optional, but offset is optional
|
||||||
|
assert(buf != NULL, "sanity check");
|
||||||
|
|
||||||
if (Decoder::decode(addr, buf, buflen, offset)) {
|
if (Decoder::decode(addr, buf, buflen, offset)) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
if (offset != NULL) *offset = -1;
|
if (offset != NULL) *offset = -1;
|
||||||
if (buf != NULL) buf[0] = '\0';
|
buf[0] = '\0';
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2689,6 +2695,19 @@ address os::win32::fast_jni_accessor_wrapper(BasicType type) {
|
|||||||
}
|
}
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifndef PRODUCT
|
||||||
|
void os::win32::call_test_func_with_wrapper(void (*funcPtr)(void)) {
|
||||||
|
// Install a win32 structured exception handler around the test
|
||||||
|
// function call so the VM can generate an error dump if needed.
|
||||||
|
__try {
|
||||||
|
(*funcPtr)();
|
||||||
|
} __except(topLevelExceptionFilter(
|
||||||
|
(_EXCEPTION_POINTERS*)_exception_info())) {
|
||||||
|
// Nothing to do.
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
// Virtual Memory
|
// Virtual Memory
|
||||||
|
|
||||||
int os::vm_page_size() { return os::win32::vm_page_size(); }
|
int os::vm_page_size() { return os::win32::vm_page_size(); }
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 1997, 2012, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -94,6 +94,10 @@ class win32 {
|
|||||||
static address fast_jni_accessor_wrapper(BasicType);
|
static address fast_jni_accessor_wrapper(BasicType);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#ifndef PRODUCT
|
||||||
|
static void call_test_func_with_wrapper(void (*funcPtr)(void));
|
||||||
|
#endif
|
||||||
|
|
||||||
// filter function to ignore faults on serializations page
|
// filter function to ignore faults on serializations page
|
||||||
static LONG WINAPI serialize_fault_filter(struct _EXCEPTION_POINTERS* e);
|
static LONG WINAPI serialize_fault_filter(struct _EXCEPTION_POINTERS* e);
|
||||||
};
|
};
|
||||||
|
@ -106,4 +106,10 @@ inline size_t os::write(int fd, const void *buf, unsigned int nBytes) {
|
|||||||
inline int os::close(int fd) {
|
inline int os::close(int fd) {
|
||||||
return ::close(fd);
|
return ::close(fd);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifndef PRODUCT
|
||||||
|
#define CALL_TEST_FUNC_WITH_WRAPPER_IF_NEEDED(f) \
|
||||||
|
os::win32::call_test_func_with_wrapper(f)
|
||||||
|
#endif
|
||||||
|
|
||||||
#endif // OS_WINDOWS_VM_OS_WINDOWS_INLINE_HPP
|
#endif // OS_WINDOWS_VM_OS_WINDOWS_INLINE_HPP
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 1997, 2012, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -146,7 +146,7 @@ public:
|
|||||||
// Public Methods
|
// Public Methods
|
||||||
Form(int formType=0, int line=0)
|
Form(int formType=0, int line=0)
|
||||||
: _next(NULL), _linenum(line), _ftype(formType) { };
|
: _next(NULL), _linenum(line), _ftype(formType) { };
|
||||||
~Form() {};
|
virtual ~Form() {};
|
||||||
|
|
||||||
virtual bool ideal_only() const {
|
virtual bool ideal_only() const {
|
||||||
assert(0,"Check of ideal status on non-instruction/operand form.\n");
|
assert(0,"Check of ideal status on non-instruction/operand form.\n");
|
||||||
|
@ -318,17 +318,17 @@ class KeepAliveVisitor : public HierarchyVisitor<KeepAliveVisitor> {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
// A method family contains a set of all methods that implement a single
|
// A method family contains a set of all methods that implement a single
|
||||||
// language-level method. Because of erasure, these methods may have different
|
// erased method. As members of the set are collected while walking over the
|
||||||
// signatures. As members of the set are collected while walking over the
|
|
||||||
// hierarchy, they are tagged with a qualification state. The qualification
|
// hierarchy, they are tagged with a qualification state. The qualification
|
||||||
// state for an erased method is set to disqualified if there exists a path
|
// state for an erased method is set to disqualified if there exists a path
|
||||||
// from the root of hierarchy to the method that contains an interleaving
|
// from the root of hierarchy to the method that contains an interleaving
|
||||||
// language-equivalent method defined in an interface.
|
// erased method defined in an interface.
|
||||||
|
|
||||||
class MethodFamily : public ResourceObj {
|
class MethodFamily : public ResourceObj {
|
||||||
private:
|
private:
|
||||||
|
|
||||||
generic::MethodDescriptor* _descriptor; // language-level description
|
|
||||||
GrowableArray<Pair<Method*,QualifiedState> > _members;
|
GrowableArray<Pair<Method*,QualifiedState> > _members;
|
||||||
ResourceHashtable<Method*, int> _member_index;
|
ResourceHashtable<Method*, int> _member_index;
|
||||||
|
|
||||||
@ -358,15 +358,8 @@ class MethodFamily : public ResourceObj {
|
|||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
MethodFamily(generic::MethodDescriptor* canonical_desc)
|
MethodFamily()
|
||||||
: _descriptor(canonical_desc), _selected_target(NULL),
|
: _selected_target(NULL), _exception_message(NULL) {}
|
||||||
_exception_message(NULL) {}
|
|
||||||
|
|
||||||
generic::MethodDescriptor* descriptor() const { return _descriptor; }
|
|
||||||
|
|
||||||
bool descriptor_matches(generic::MethodDescriptor* md, generic::Context* ctx) {
|
|
||||||
return descriptor()->covariant_match(md, ctx);
|
|
||||||
}
|
|
||||||
|
|
||||||
void set_target_if_empty(Method* m) {
|
void set_target_if_empty(Method* m) {
|
||||||
if (_selected_target == NULL && !m->is_overpass()) {
|
if (_selected_target == NULL && !m->is_overpass()) {
|
||||||
@ -441,16 +434,10 @@ class MethodFamily : public ResourceObj {
|
|||||||
}
|
}
|
||||||
|
|
||||||
#ifndef PRODUCT
|
#ifndef PRODUCT
|
||||||
void print_on(outputStream* str) const {
|
void print_sig_on(outputStream* str, Symbol* signature, int indent) const {
|
||||||
print_on(str, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
void print_on(outputStream* str, int indent) const {
|
|
||||||
streamIndentor si(str, indent * 2);
|
streamIndentor si(str, indent * 2);
|
||||||
|
|
||||||
generic::Context ctx(NULL); // empty, as _descriptor already canonicalized
|
str->indent().print_cr("Logical Method %s:", signature->as_C_string());
|
||||||
TempNewSymbol family = descriptor()->reify_signature(&ctx, Thread::current());
|
|
||||||
str->indent().print_cr("Logical Method %s:", family->as_C_string());
|
|
||||||
|
|
||||||
streamIndentor si2(str);
|
streamIndentor si2(str);
|
||||||
for (int i = 0; i < _members.length(); ++i) {
|
for (int i = 0; i < _members.length(); ++i) {
|
||||||
@ -516,38 +503,94 @@ Symbol* MethodFamily::generate_conflicts_message(GrowableArray<Method*>* methods
|
|||||||
return SymbolTable::new_symbol(ss.base(), (int)ss.size(), CHECK_NULL);
|
return SymbolTable::new_symbol(ss.base(), (int)ss.size(), CHECK_NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// A generic method family contains a set of all methods that implement a single
|
||||||
|
// language-level method. Because of erasure, these methods may have different
|
||||||
|
// signatures. As members of the set are collected while walking over the
|
||||||
|
// hierarchy, they are tagged with a qualification state. The qualification
|
||||||
|
// state for an erased method is set to disqualified if there exists a path
|
||||||
|
// from the root of hierarchy to the method that contains an interleaving
|
||||||
|
// language-equivalent method defined in an interface.
|
||||||
|
class GenericMethodFamily : public MethodFamily {
|
||||||
|
private:
|
||||||
|
|
||||||
|
generic::MethodDescriptor* _descriptor; // language-level description
|
||||||
|
|
||||||
|
public:
|
||||||
|
|
||||||
|
GenericMethodFamily(generic::MethodDescriptor* canonical_desc)
|
||||||
|
: _descriptor(canonical_desc) {}
|
||||||
|
|
||||||
|
generic::MethodDescriptor* descriptor() const { return _descriptor; }
|
||||||
|
|
||||||
|
bool descriptor_matches(generic::MethodDescriptor* md, generic::Context* ctx) {
|
||||||
|
return descriptor()->covariant_match(md, ctx);
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifndef PRODUCT
|
||||||
|
Symbol* get_generic_sig() const {
|
||||||
|
|
||||||
|
generic::Context ctx(NULL); // empty, as _descriptor already canonicalized
|
||||||
|
TempNewSymbol sig = descriptor()->reify_signature(&ctx, Thread::current());
|
||||||
|
return sig;
|
||||||
|
}
|
||||||
|
#endif // ndef PRODUCT
|
||||||
|
};
|
||||||
|
|
||||||
class StateRestorer;
|
class StateRestorer;
|
||||||
|
|
||||||
// StatefulMethodFamily is a wrapper around MethodFamily that maintains the
|
// StatefulMethodFamily is a wrapper around a MethodFamily that maintains the
|
||||||
// qualification state during hierarchy visitation, and applies that state
|
// qualification state during hierarchy visitation, and applies that state
|
||||||
// when adding members to the MethodFamily.
|
// when adding members to the MethodFamily
|
||||||
class StatefulMethodFamily : public ResourceObj {
|
class StatefulMethodFamily : public ResourceObj {
|
||||||
friend class StateRestorer;
|
friend class StateRestorer;
|
||||||
private:
|
private:
|
||||||
MethodFamily* _method;
|
|
||||||
QualifiedState _qualification_state;
|
QualifiedState _qualification_state;
|
||||||
|
|
||||||
void set_qualification_state(QualifiedState state) {
|
void set_qualification_state(QualifiedState state) {
|
||||||
_qualification_state = state;
|
_qualification_state = state;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
MethodFamily* _method_family;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
StatefulMethodFamily(generic::MethodDescriptor* md, generic::Context* ctx) {
|
StatefulMethodFamily() {
|
||||||
_method = new MethodFamily(md->canonicalize(ctx));
|
_method_family = new MethodFamily();
|
||||||
_qualification_state = QUALIFIED;
|
_qualification_state = QUALIFIED;
|
||||||
}
|
}
|
||||||
|
|
||||||
void set_target_if_empty(Method* m) { _method->set_target_if_empty(m); }
|
StatefulMethodFamily(MethodFamily* mf) {
|
||||||
|
_method_family = mf;
|
||||||
MethodFamily* get_method_family() { return _method; }
|
_qualification_state = QUALIFIED;
|
||||||
|
|
||||||
bool descriptor_matches(generic::MethodDescriptor* md, generic::Context* ctx) {
|
|
||||||
return _method->descriptor_matches(md, ctx);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void set_target_if_empty(Method* m) { _method_family->set_target_if_empty(m); }
|
||||||
|
|
||||||
|
MethodFamily* get_method_family() { return _method_family; }
|
||||||
|
|
||||||
StateRestorer* record_method_and_dq_further(Method* mo);
|
StateRestorer* record_method_and_dq_further(Method* mo);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
||||||
|
// StatefulGenericMethodFamily is a wrapper around GenericMethodFamily that maintains the
|
||||||
|
// qualification state during hierarchy visitation, and applies that state
|
||||||
|
// when adding members to the GenericMethodFamily.
|
||||||
|
class StatefulGenericMethodFamily : public StatefulMethodFamily {
|
||||||
|
|
||||||
|
public:
|
||||||
|
StatefulGenericMethodFamily(generic::MethodDescriptor* md, generic::Context* ctx)
|
||||||
|
: StatefulMethodFamily(new GenericMethodFamily(md->canonicalize(ctx))) {
|
||||||
|
|
||||||
|
}
|
||||||
|
GenericMethodFamily* get_method_family() {
|
||||||
|
return (GenericMethodFamily*)_method_family;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool descriptor_matches(generic::MethodDescriptor* md, generic::Context* ctx) {
|
||||||
|
return get_method_family()->descriptor_matches(md, ctx);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
class StateRestorer : public PseudoScopeMark {
|
class StateRestorer : public PseudoScopeMark {
|
||||||
private:
|
private:
|
||||||
StatefulMethodFamily* _method;
|
StatefulMethodFamily* _method;
|
||||||
@ -563,9 +606,9 @@ class StateRestorer : public PseudoScopeMark {
|
|||||||
StateRestorer* StatefulMethodFamily::record_method_and_dq_further(Method* mo) {
|
StateRestorer* StatefulMethodFamily::record_method_and_dq_further(Method* mo) {
|
||||||
StateRestorer* mark = new StateRestorer(this, _qualification_state);
|
StateRestorer* mark = new StateRestorer(this, _qualification_state);
|
||||||
if (_qualification_state == QUALIFIED) {
|
if (_qualification_state == QUALIFIED) {
|
||||||
_method->record_qualified_method(mo);
|
_method_family->record_qualified_method(mo);
|
||||||
} else {
|
} else {
|
||||||
_method->record_disqualified_method(mo);
|
_method_family->record_disqualified_method(mo);
|
||||||
}
|
}
|
||||||
// Everything found "above"??? this method in the hierarchy walk is set to
|
// Everything found "above"??? this method in the hierarchy walk is set to
|
||||||
// disqualified
|
// disqualified
|
||||||
@ -573,15 +616,15 @@ StateRestorer* StatefulMethodFamily::record_method_and_dq_further(Method* mo) {
|
|||||||
return mark;
|
return mark;
|
||||||
}
|
}
|
||||||
|
|
||||||
class StatefulMethodFamilies : public ResourceObj {
|
class StatefulGenericMethodFamilies : public ResourceObj {
|
||||||
private:
|
private:
|
||||||
GrowableArray<StatefulMethodFamily*> _methods;
|
GrowableArray<StatefulGenericMethodFamily*> _methods;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
StatefulMethodFamily* find_matching(
|
StatefulGenericMethodFamily* find_matching(
|
||||||
generic::MethodDescriptor* md, generic::Context* ctx) {
|
generic::MethodDescriptor* md, generic::Context* ctx) {
|
||||||
for (int i = 0; i < _methods.length(); ++i) {
|
for (int i = 0; i < _methods.length(); ++i) {
|
||||||
StatefulMethodFamily* existing = _methods.at(i);
|
StatefulGenericMethodFamily* existing = _methods.at(i);
|
||||||
if (existing->descriptor_matches(md, ctx)) {
|
if (existing->descriptor_matches(md, ctx)) {
|
||||||
return existing;
|
return existing;
|
||||||
}
|
}
|
||||||
@ -589,17 +632,17 @@ class StatefulMethodFamilies : public ResourceObj {
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
StatefulMethodFamily* find_matching_or_create(
|
StatefulGenericMethodFamily* find_matching_or_create(
|
||||||
generic::MethodDescriptor* md, generic::Context* ctx) {
|
generic::MethodDescriptor* md, generic::Context* ctx) {
|
||||||
StatefulMethodFamily* method = find_matching(md, ctx);
|
StatefulGenericMethodFamily* method = find_matching(md, ctx);
|
||||||
if (method == NULL) {
|
if (method == NULL) {
|
||||||
method = new StatefulMethodFamily(md, ctx);
|
method = new StatefulGenericMethodFamily(md, ctx);
|
||||||
_methods.append(method);
|
_methods.append(method);
|
||||||
}
|
}
|
||||||
return method;
|
return method;
|
||||||
}
|
}
|
||||||
|
|
||||||
void extract_families_into(GrowableArray<MethodFamily*>* array) {
|
void extract_families_into(GrowableArray<GenericMethodFamily*>* array) {
|
||||||
for (int i = 0; i < _methods.length(); ++i) {
|
for (int i = 0; i < _methods.length(); ++i) {
|
||||||
array->append(_methods.at(i)->get_method_family());
|
array->append(_methods.at(i)->get_method_family());
|
||||||
}
|
}
|
||||||
@ -683,26 +726,79 @@ static GrowableArray<EmptyVtableSlot*>* find_empty_vtable_slots(
|
|||||||
return slots;
|
return slots;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Iterates over the superinterface type hierarchy looking for all methods
|
||||||
|
// with a specific erased signature.
|
||||||
|
class FindMethodsByErasedSig : public HierarchyVisitor<FindMethodsByErasedSig> {
|
||||||
|
private:
|
||||||
|
// Context data
|
||||||
|
Symbol* _method_name;
|
||||||
|
Symbol* _method_signature;
|
||||||
|
StatefulMethodFamily* _family;
|
||||||
|
|
||||||
|
public:
|
||||||
|
FindMethodsByErasedSig(Symbol* name, Symbol* signature) :
|
||||||
|
_method_name(name), _method_signature(signature),
|
||||||
|
_family(NULL) {}
|
||||||
|
|
||||||
|
void get_discovered_family(MethodFamily** family) {
|
||||||
|
if (_family != NULL) {
|
||||||
|
*family = _family->get_method_family();
|
||||||
|
} else {
|
||||||
|
*family = NULL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void* new_node_data(InstanceKlass* cls) { return new PseudoScope(); }
|
||||||
|
void free_node_data(void* node_data) {
|
||||||
|
PseudoScope::cast(node_data)->destroy();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Find all methods on this hierarchy that match this
|
||||||
|
// method's erased (name, signature)
|
||||||
|
bool visit() {
|
||||||
|
PseudoScope* scope = PseudoScope::cast(current_data());
|
||||||
|
InstanceKlass* iklass = current_class();
|
||||||
|
|
||||||
|
Method* m = iklass->find_method(_method_name, _method_signature);
|
||||||
|
if (m != NULL) {
|
||||||
|
if (_family == NULL) {
|
||||||
|
_family = new StatefulMethodFamily();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (iklass->is_interface()) {
|
||||||
|
StateRestorer* restorer = _family->record_method_and_dq_further(m);
|
||||||
|
scope->add_mark(restorer);
|
||||||
|
} else {
|
||||||
|
// This is the rule that methods in classes "win" (bad word) over
|
||||||
|
// methods in interfaces. This works because of single inheritance
|
||||||
|
_family->set_target_if_empty(m);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
// Iterates over the type hierarchy looking for all methods with a specific
|
// Iterates over the type hierarchy looking for all methods with a specific
|
||||||
// method name. The result of this is a set of method families each of
|
// method name. The result of this is a set of method families each of
|
||||||
// which is populated with a set of methods that implement the same
|
// which is populated with a set of methods that implement the same
|
||||||
// language-level signature.
|
// language-level signature.
|
||||||
class FindMethodsByName : public HierarchyVisitor<FindMethodsByName> {
|
class FindMethodsByGenericSig : public HierarchyVisitor<FindMethodsByGenericSig> {
|
||||||
private:
|
private:
|
||||||
// Context data
|
// Context data
|
||||||
Thread* THREAD;
|
Thread* THREAD;
|
||||||
generic::DescriptorCache* _cache;
|
generic::DescriptorCache* _cache;
|
||||||
Symbol* _method_name;
|
Symbol* _method_name;
|
||||||
generic::Context* _ctx;
|
generic::Context* _ctx;
|
||||||
StatefulMethodFamilies _families;
|
StatefulGenericMethodFamilies _families;
|
||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
FindMethodsByName(generic::DescriptorCache* cache, Symbol* name,
|
FindMethodsByGenericSig(generic::DescriptorCache* cache, Symbol* name,
|
||||||
generic::Context* ctx, Thread* thread) :
|
generic::Context* ctx, Thread* thread) :
|
||||||
_cache(cache), _method_name(name), _ctx(ctx), THREAD(thread) {}
|
_cache(cache), _method_name(name), _ctx(ctx), THREAD(thread) {}
|
||||||
|
|
||||||
void get_discovered_families(GrowableArray<MethodFamily*>* methods) {
|
void get_discovered_families(GrowableArray<GenericMethodFamily*>* methods) {
|
||||||
_families.extract_families_into(methods);
|
_families.extract_families_into(methods);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -733,7 +829,7 @@ class FindMethodsByName : public HierarchyVisitor<FindMethodsByName> {
|
|||||||
// Find all methods on this hierarchy that match this method
|
// Find all methods on this hierarchy that match this method
|
||||||
// (name, signature). This class collects other families of this
|
// (name, signature). This class collects other families of this
|
||||||
// method name.
|
// method name.
|
||||||
StatefulMethodFamily* family =
|
StatefulGenericMethodFamily* family =
|
||||||
_families.find_matching_or_create(md, _ctx);
|
_families.find_matching_or_create(md, _ctx);
|
||||||
|
|
||||||
if (klass->is_interface()) {
|
if (klass->is_interface()) {
|
||||||
@ -752,8 +848,8 @@ class FindMethodsByName : public HierarchyVisitor<FindMethodsByName> {
|
|||||||
};
|
};
|
||||||
|
|
||||||
#ifndef PRODUCT
|
#ifndef PRODUCT
|
||||||
static void print_families(
|
static void print_generic_families(
|
||||||
GrowableArray<MethodFamily*>* methods, Symbol* match) {
|
GrowableArray<GenericMethodFamily*>* methods, Symbol* match) {
|
||||||
streamIndentor si(tty, 4);
|
streamIndentor si(tty, 4);
|
||||||
if (methods->length() == 0) {
|
if (methods->length() == 0) {
|
||||||
tty->indent();
|
tty->indent();
|
||||||
@ -761,22 +857,87 @@ static void print_families(
|
|||||||
}
|
}
|
||||||
for (int i = 0; i < methods->length(); ++i) {
|
for (int i = 0; i < methods->length(); ++i) {
|
||||||
tty->indent();
|
tty->indent();
|
||||||
MethodFamily* lm = methods->at(i);
|
GenericMethodFamily* lm = methods->at(i);
|
||||||
if (lm->contains_signature(match)) {
|
if (lm->contains_signature(match)) {
|
||||||
tty->print_cr("<Matching>");
|
tty->print_cr("<Matching>");
|
||||||
} else {
|
} else {
|
||||||
tty->print_cr("<Non-Matching>");
|
tty->print_cr("<Non-Matching>");
|
||||||
}
|
}
|
||||||
lm->print_on(tty, 1);
|
lm->print_sig_on(tty, lm->get_generic_sig(), 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif // ndef PRODUCT
|
#endif // ndef PRODUCT
|
||||||
|
|
||||||
static void merge_in_new_methods(InstanceKlass* klass,
|
|
||||||
GrowableArray<Method*>* new_methods, TRAPS);
|
|
||||||
static void create_overpasses(
|
static void create_overpasses(
|
||||||
GrowableArray<EmptyVtableSlot*>* slots, InstanceKlass* klass, TRAPS);
|
GrowableArray<EmptyVtableSlot*>* slots, InstanceKlass* klass, TRAPS);
|
||||||
|
|
||||||
|
static void generate_generic_defaults(
|
||||||
|
InstanceKlass* klass, GrowableArray<EmptyVtableSlot*>* empty_slots,
|
||||||
|
EmptyVtableSlot* slot, int current_slot_index, TRAPS) {
|
||||||
|
|
||||||
|
if (slot->is_bound()) {
|
||||||
|
#ifndef PRODUCT
|
||||||
|
if (TraceDefaultMethods) {
|
||||||
|
streamIndentor si(tty, 4);
|
||||||
|
tty->indent().print_cr("Already bound to logical method:");
|
||||||
|
GenericMethodFamily* lm = (GenericMethodFamily*)(slot->get_binding());
|
||||||
|
lm->print_sig_on(tty, lm->get_generic_sig(), 1);
|
||||||
|
}
|
||||||
|
#endif // ndef PRODUCT
|
||||||
|
return; // covered by previous processing
|
||||||
|
}
|
||||||
|
|
||||||
|
generic::DescriptorCache cache;
|
||||||
|
|
||||||
|
generic::Context ctx(&cache);
|
||||||
|
FindMethodsByGenericSig visitor(&cache, slot->name(), &ctx, CHECK);
|
||||||
|
visitor.run(klass);
|
||||||
|
|
||||||
|
GrowableArray<GenericMethodFamily*> discovered_families;
|
||||||
|
visitor.get_discovered_families(&discovered_families);
|
||||||
|
|
||||||
|
#ifndef PRODUCT
|
||||||
|
if (TraceDefaultMethods) {
|
||||||
|
print_generic_families(&discovered_families, slot->signature());
|
||||||
|
}
|
||||||
|
#endif // ndef PRODUCT
|
||||||
|
|
||||||
|
// Find and populate any other slots that match the discovered families
|
||||||
|
for (int j = current_slot_index; j < empty_slots->length(); ++j) {
|
||||||
|
EmptyVtableSlot* open_slot = empty_slots->at(j);
|
||||||
|
|
||||||
|
if (slot->name() == open_slot->name()) {
|
||||||
|
for (int k = 0; k < discovered_families.length(); ++k) {
|
||||||
|
GenericMethodFamily* lm = discovered_families.at(k);
|
||||||
|
|
||||||
|
if (lm->contains_signature(open_slot->signature())) {
|
||||||
|
lm->determine_target(klass, CHECK);
|
||||||
|
open_slot->bind_family(lm);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void generate_erased_defaults(
|
||||||
|
InstanceKlass* klass, GrowableArray<EmptyVtableSlot*>* empty_slots,
|
||||||
|
EmptyVtableSlot* slot, TRAPS) {
|
||||||
|
|
||||||
|
// sets up a set of methods with the same exact erased signature
|
||||||
|
FindMethodsByErasedSig visitor(slot->name(), slot->signature());
|
||||||
|
visitor.run(klass);
|
||||||
|
|
||||||
|
MethodFamily* family;
|
||||||
|
visitor.get_discovered_family(&family);
|
||||||
|
if (family != NULL) {
|
||||||
|
family->determine_target(klass, CHECK);
|
||||||
|
slot->bind_family(family);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
static void merge_in_new_methods(InstanceKlass* klass,
|
||||||
|
GrowableArray<Method*>* new_methods, TRAPS);
|
||||||
|
|
||||||
// This is the guts of the default methods implementation. This is called just
|
// This is the guts of the default methods implementation. This is called just
|
||||||
// after the classfile has been parsed if some ancestor has default methods.
|
// after the classfile has been parsed if some ancestor has default methods.
|
||||||
//
|
//
|
||||||
@ -807,8 +968,6 @@ void DefaultMethods::generate_default_methods(
|
|||||||
// whatever scope it's in.
|
// whatever scope it's in.
|
||||||
ResourceMark rm(THREAD);
|
ResourceMark rm(THREAD);
|
||||||
|
|
||||||
generic::DescriptorCache cache;
|
|
||||||
|
|
||||||
// Keep entire hierarchy alive for the duration of the computation
|
// Keep entire hierarchy alive for the duration of the computation
|
||||||
KeepAliveRegistrar keepAlive(THREAD);
|
KeepAliveRegistrar keepAlive(THREAD);
|
||||||
KeepAliveVisitor loadKeepAlive(&keepAlive);
|
KeepAliveVisitor loadKeepAlive(&keepAlive);
|
||||||
@ -837,47 +996,13 @@ void DefaultMethods::generate_default_methods(
|
|||||||
tty->print_cr("");
|
tty->print_cr("");
|
||||||
}
|
}
|
||||||
#endif // ndef PRODUCT
|
#endif // ndef PRODUCT
|
||||||
if (slot->is_bound()) {
|
|
||||||
#ifndef PRODUCT
|
|
||||||
if (TraceDefaultMethods) {
|
|
||||||
streamIndentor si(tty, 4);
|
|
||||||
tty->indent().print_cr("Already bound to logical method:");
|
|
||||||
slot->get_binding()->print_on(tty, 1);
|
|
||||||
}
|
|
||||||
#endif // ndef PRODUCT
|
|
||||||
continue; // covered by previous processing
|
|
||||||
}
|
|
||||||
|
|
||||||
generic::Context ctx(&cache);
|
if (ParseGenericDefaults) {
|
||||||
FindMethodsByName visitor(&cache, slot->name(), &ctx, CHECK);
|
generate_generic_defaults(klass, empty_slots, slot, i, CHECK);
|
||||||
visitor.run(klass);
|
} else {
|
||||||
|
generate_erased_defaults(klass, empty_slots, slot, CHECK);
|
||||||
GrowableArray<MethodFamily*> discovered_families;
|
|
||||||
visitor.get_discovered_families(&discovered_families);
|
|
||||||
|
|
||||||
#ifndef PRODUCT
|
|
||||||
if (TraceDefaultMethods) {
|
|
||||||
print_families(&discovered_families, slot->signature());
|
|
||||||
}
|
|
||||||
#endif // ndef PRODUCT
|
|
||||||
|
|
||||||
// Find and populate any other slots that match the discovered families
|
|
||||||
for (int j = i; j < empty_slots->length(); ++j) {
|
|
||||||
EmptyVtableSlot* open_slot = empty_slots->at(j);
|
|
||||||
|
|
||||||
if (slot->name() == open_slot->name()) {
|
|
||||||
for (int k = 0; k < discovered_families.length(); ++k) {
|
|
||||||
MethodFamily* lm = discovered_families.at(k);
|
|
||||||
|
|
||||||
if (lm->contains_signature(open_slot->signature())) {
|
|
||||||
lm->determine_target(klass, CHECK);
|
|
||||||
open_slot->bind_family(lm);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifndef PRODUCT
|
#ifndef PRODUCT
|
||||||
if (TraceDefaultMethods) {
|
if (TraceDefaultMethods) {
|
||||||
tty->print_cr("Creating overpasses...");
|
tty->print_cr("Creating overpasses...");
|
||||||
@ -893,7 +1018,6 @@ void DefaultMethods::generate_default_methods(
|
|||||||
#endif // ndef PRODUCT
|
#endif // ndef PRODUCT
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Generic analysis was used upon interface '_target' and found a unique
|
* Generic analysis was used upon interface '_target' and found a unique
|
||||||
* default method candidate with generic signature '_method_desc'. This
|
* default method candidate with generic signature '_method_desc'. This
|
||||||
@ -912,17 +1036,85 @@ void DefaultMethods::generate_default_methods(
|
|||||||
* the selected method along that path.
|
* the selected method along that path.
|
||||||
*/
|
*/
|
||||||
class ShadowChecker : public HierarchyVisitor<ShadowChecker> {
|
class ShadowChecker : public HierarchyVisitor<ShadowChecker> {
|
||||||
private:
|
protected:
|
||||||
generic::DescriptorCache* _cache;
|
|
||||||
Thread* THREAD;
|
Thread* THREAD;
|
||||||
|
|
||||||
InstanceKlass* _target;
|
InstanceKlass* _target;
|
||||||
|
|
||||||
Symbol* _method_name;
|
Symbol* _method_name;
|
||||||
InstanceKlass* _method_holder;
|
InstanceKlass* _method_holder;
|
||||||
generic::MethodDescriptor* _method_desc;
|
|
||||||
bool _found_shadow;
|
bool _found_shadow;
|
||||||
|
|
||||||
|
|
||||||
|
public:
|
||||||
|
|
||||||
|
ShadowChecker(Thread* thread, Symbol* name, InstanceKlass* holder,
|
||||||
|
InstanceKlass* target)
|
||||||
|
: THREAD(thread), _method_name(name), _method_holder(holder),
|
||||||
|
_target(target), _found_shadow(false) {}
|
||||||
|
|
||||||
|
void* new_node_data(InstanceKlass* cls) { return NULL; }
|
||||||
|
void free_node_data(void* data) { return; }
|
||||||
|
|
||||||
|
bool visit() {
|
||||||
|
InstanceKlass* ik = current_class();
|
||||||
|
if (ik == _target && current_depth() == 1) {
|
||||||
|
return false; // This was the specified super -- no need to search it
|
||||||
|
}
|
||||||
|
if (ik == _method_holder || ik == _target) {
|
||||||
|
// We found a path that should be examined to see if it shadows _method
|
||||||
|
if (path_has_shadow()) {
|
||||||
|
_found_shadow = true;
|
||||||
|
cancel_iteration();
|
||||||
|
}
|
||||||
|
return false; // no need to continue up hierarchy
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
virtual bool path_has_shadow() = 0;
|
||||||
|
bool found_shadow() { return _found_shadow; }
|
||||||
|
};
|
||||||
|
|
||||||
|
// Used for Invokespecial.
|
||||||
|
// Invokespecial is allowed to invoke a concrete interface method
|
||||||
|
// and can be used to disambuiguate among qualified candidates,
|
||||||
|
// which are methods in immediate superinterfaces,
|
||||||
|
// but may not be used to invoke a candidate that would be shadowed
|
||||||
|
// from the perspective of the caller.
|
||||||
|
// Invokespecial is also used in the overpass generation today
|
||||||
|
// We re-run the shadowchecker because we can't distinguish this case,
|
||||||
|
// but it should return the same answer, since the overpass target
|
||||||
|
// is now the invokespecial caller.
|
||||||
|
class ErasedShadowChecker : public ShadowChecker {
|
||||||
|
private:
|
||||||
|
bool path_has_shadow() {
|
||||||
|
|
||||||
|
for (int i = current_depth() - 1; i > 0; --i) {
|
||||||
|
InstanceKlass* ik = class_at_depth(i);
|
||||||
|
|
||||||
|
if (ik->is_interface()) {
|
||||||
|
int end;
|
||||||
|
int start = ik->find_method_by_name(_method_name, &end);
|
||||||
|
if (start != -1) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
public:
|
||||||
|
|
||||||
|
ErasedShadowChecker(Thread* thread, Symbol* name, InstanceKlass* holder,
|
||||||
|
InstanceKlass* target)
|
||||||
|
: ShadowChecker(thread, name, holder, target) {}
|
||||||
|
};
|
||||||
|
|
||||||
|
class GenericShadowChecker : public ShadowChecker {
|
||||||
|
private:
|
||||||
|
generic::DescriptorCache* _cache;
|
||||||
|
generic::MethodDescriptor* _method_desc;
|
||||||
|
|
||||||
bool path_has_shadow() {
|
bool path_has_shadow() {
|
||||||
generic::Context ctx(_cache);
|
generic::Context ctx(_cache);
|
||||||
|
|
||||||
@ -950,104 +1142,42 @@ class ShadowChecker : public HierarchyVisitor<ShadowChecker> {
|
|||||||
|
|
||||||
public:
|
public:
|
||||||
|
|
||||||
ShadowChecker(generic::DescriptorCache* cache, Thread* thread,
|
GenericShadowChecker(generic::DescriptorCache* cache, Thread* thread,
|
||||||
Symbol* name, InstanceKlass* holder, generic::MethodDescriptor* desc,
|
Symbol* name, InstanceKlass* holder, generic::MethodDescriptor* desc,
|
||||||
InstanceKlass* target)
|
InstanceKlass* target)
|
||||||
: _cache(cache), THREAD(thread), _method_name(name), _method_holder(holder),
|
: ShadowChecker(thread, name, holder, target) {
|
||||||
_method_desc(desc), _target(target), _found_shadow(false) {}
|
_cache = cache;
|
||||||
|
_method_desc = desc;
|
||||||
void* new_node_data(InstanceKlass* cls) { return NULL; }
|
|
||||||
void free_node_data(void* data) { return; }
|
|
||||||
|
|
||||||
bool visit() {
|
|
||||||
InstanceKlass* ik = current_class();
|
|
||||||
if (ik == _target && current_depth() == 1) {
|
|
||||||
return false; // This was the specified super -- no need to search it
|
|
||||||
}
|
}
|
||||||
if (ik == _method_holder || ik == _target) {
|
|
||||||
// We found a path that should be examined to see if it shadows _method
|
|
||||||
if (path_has_shadow()) {
|
|
||||||
_found_shadow = true;
|
|
||||||
cancel_iteration();
|
|
||||||
}
|
|
||||||
return false; // no need to continue up hierarchy
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
bool found_shadow() { return _found_shadow; }
|
|
||||||
};
|
};
|
||||||
|
|
||||||
// This is called during linktime when we find an invokespecial call that
|
|
||||||
// refers to a direct superinterface. It indicates that we should find the
|
|
||||||
// default method in the hierarchy of that superinterface, and if that method
|
|
||||||
// would have been a candidate from the point of view of 'this' class, then we
|
|
||||||
// return that method.
|
|
||||||
Method* DefaultMethods::find_super_default(
|
|
||||||
Klass* cls, Klass* super, Symbol* method_name, Symbol* sig, TRAPS) {
|
|
||||||
|
|
||||||
ResourceMark rm(THREAD);
|
|
||||||
|
|
||||||
assert(cls != NULL && super != NULL, "Need real classes");
|
// Find the unique qualified candidate from the perspective of the super_class
|
||||||
|
// which is the resolved_klass, which must be an immediate superinterface
|
||||||
|
// of klass
|
||||||
|
Method* find_erased_super_default(InstanceKlass* current_class, InstanceKlass* super_class, Symbol* method_name, Symbol* sig, TRAPS) {
|
||||||
|
|
||||||
InstanceKlass* current_class = InstanceKlass::cast(cls);
|
FindMethodsByErasedSig visitor(method_name, sig);
|
||||||
InstanceKlass* direction = InstanceKlass::cast(super);
|
visitor.run(super_class); // find candidates from resolved_klass
|
||||||
|
|
||||||
// Keep entire hierarchy alive for the duration of the computation
|
MethodFamily* family;
|
||||||
KeepAliveRegistrar keepAlive(THREAD);
|
visitor.get_discovered_family(&family);
|
||||||
KeepAliveVisitor loadKeepAlive(&keepAlive);
|
|
||||||
loadKeepAlive.run(current_class);
|
|
||||||
|
|
||||||
#ifndef PRODUCT
|
if (family != NULL) {
|
||||||
if (TraceDefaultMethods) {
|
family->determine_target(current_class, CHECK_NULL); // get target from current_class
|
||||||
tty->print_cr("Finding super default method %s.%s%s from %s",
|
|
||||||
direction->name()->as_C_string(),
|
|
||||||
method_name->as_C_string(), sig->as_C_string(),
|
|
||||||
current_class->name()->as_C_string());
|
|
||||||
}
|
|
||||||
#endif // ndef PRODUCT
|
|
||||||
|
|
||||||
if (!direction->is_interface()) {
|
|
||||||
// We should not be here
|
|
||||||
return NULL;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
generic::DescriptorCache cache;
|
if (family->has_target()) {
|
||||||
generic::Context ctx(&cache);
|
Method* target = family->get_selected_target();
|
||||||
|
|
||||||
// Prime the initial generic context for current -> direction
|
|
||||||
ctx.apply_type_arguments(current_class, direction, CHECK_NULL);
|
|
||||||
|
|
||||||
FindMethodsByName visitor(&cache, method_name, &ctx, CHECK_NULL);
|
|
||||||
visitor.run(direction);
|
|
||||||
|
|
||||||
GrowableArray<MethodFamily*> families;
|
|
||||||
visitor.get_discovered_families(&families);
|
|
||||||
|
|
||||||
#ifndef PRODUCT
|
|
||||||
if (TraceDefaultMethods) {
|
|
||||||
print_families(&families, sig);
|
|
||||||
}
|
|
||||||
#endif // ndef PRODUCT
|
|
||||||
|
|
||||||
MethodFamily* selected_family = NULL;
|
|
||||||
|
|
||||||
for (int i = 0; i < families.length(); ++i) {
|
|
||||||
MethodFamily* lm = families.at(i);
|
|
||||||
if (lm->contains_signature(sig)) {
|
|
||||||
lm->determine_target(current_class, CHECK_NULL);
|
|
||||||
selected_family = lm;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (selected_family->has_target()) {
|
|
||||||
Method* target = selected_family->get_selected_target();
|
|
||||||
InstanceKlass* holder = InstanceKlass::cast(target->method_holder());
|
InstanceKlass* holder = InstanceKlass::cast(target->method_holder());
|
||||||
|
|
||||||
// Verify that the identified method is valid from the context of
|
// Verify that the identified method is valid from the context of
|
||||||
// the current class
|
// the current class, which is the caller class for invokespecial
|
||||||
ShadowChecker checker(&cache, THREAD, target->name(),
|
// link resolution, i.e. ensure there it is not shadowed.
|
||||||
holder, selected_family->descriptor(), direction);
|
// You can use invokespecial to disambiguate interface methods, but
|
||||||
|
// you can not use it to skip over an interface method that would shadow it.
|
||||||
|
ErasedShadowChecker checker(THREAD, target->name(), holder, super_class);
|
||||||
checker.run(current_class);
|
checker.run(current_class);
|
||||||
|
|
||||||
if (checker.found_shadow()) {
|
if (checker.found_shadow()) {
|
||||||
@ -1061,13 +1191,71 @@ Method* DefaultMethods::find_super_default(
|
|||||||
} else {
|
} else {
|
||||||
#ifndef PRODUCT
|
#ifndef PRODUCT
|
||||||
if (TraceDefaultMethods) {
|
if (TraceDefaultMethods) {
|
||||||
tty->print(" Returning ");
|
family->print_sig_on(tty, target->signature(), 1);
|
||||||
print_method(tty, target, true);
|
|
||||||
tty->print_cr("");
|
|
||||||
}
|
}
|
||||||
#endif // ndef PRODUCT
|
#endif // ndef PRODUCT
|
||||||
return target;
|
return target;
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
assert(family->throws_exception(), "must have target or throw");
|
||||||
|
THROW_MSG_(vmSymbols::java_lang_AbstractMethodError(),
|
||||||
|
family->get_exception_message()->as_C_string(), NULL);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// super_class is assumed to be the direct super of current_class
|
||||||
|
Method* find_generic_super_default( InstanceKlass* current_class,
|
||||||
|
InstanceKlass* super_class,
|
||||||
|
Symbol* method_name, Symbol* sig, TRAPS) {
|
||||||
|
generic::DescriptorCache cache;
|
||||||
|
generic::Context ctx(&cache);
|
||||||
|
|
||||||
|
// Prime the initial generic context for current -> super_class
|
||||||
|
ctx.apply_type_arguments(current_class, super_class, CHECK_NULL);
|
||||||
|
|
||||||
|
FindMethodsByGenericSig visitor(&cache, method_name, &ctx, CHECK_NULL);
|
||||||
|
visitor.run(super_class);
|
||||||
|
|
||||||
|
GrowableArray<GenericMethodFamily*> families;
|
||||||
|
visitor.get_discovered_families(&families);
|
||||||
|
|
||||||
|
#ifndef PRODUCT
|
||||||
|
if (TraceDefaultMethods) {
|
||||||
|
print_generic_families(&families, sig);
|
||||||
|
}
|
||||||
|
#endif // ndef PRODUCT
|
||||||
|
|
||||||
|
GenericMethodFamily* selected_family = NULL;
|
||||||
|
|
||||||
|
for (int i = 0; i < families.length(); ++i) {
|
||||||
|
GenericMethodFamily* lm = families.at(i);
|
||||||
|
if (lm->contains_signature(sig)) {
|
||||||
|
lm->determine_target(current_class, CHECK_NULL);
|
||||||
|
selected_family = lm;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (selected_family->has_target()) {
|
||||||
|
Method* target = selected_family->get_selected_target();
|
||||||
|
InstanceKlass* holder = InstanceKlass::cast(target->method_holder());
|
||||||
|
|
||||||
|
// Verify that the identified method is valid from the context of
|
||||||
|
// the current class
|
||||||
|
GenericShadowChecker checker(&cache, THREAD, target->name(),
|
||||||
|
holder, selected_family->descriptor(), super_class);
|
||||||
|
checker.run(current_class);
|
||||||
|
|
||||||
|
if (checker.found_shadow()) {
|
||||||
|
#ifndef PRODUCT
|
||||||
|
if (TraceDefaultMethods) {
|
||||||
|
tty->print_cr(" Only candidate found was shadowed.");
|
||||||
|
}
|
||||||
|
#endif // ndef PRODUCT
|
||||||
|
THROW_MSG_(vmSymbols::java_lang_AbstractMethodError(),
|
||||||
|
"Accessible default method not found", NULL);
|
||||||
|
} else {
|
||||||
|
return target;
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
assert(selected_family->throws_exception(), "must have target or throw");
|
assert(selected_family->throws_exception(), "must have target or throw");
|
||||||
THROW_MSG_(vmSymbols::java_lang_AbstractMethodError(),
|
THROW_MSG_(vmSymbols::java_lang_AbstractMethodError(),
|
||||||
@ -1075,6 +1263,71 @@ Method* DefaultMethods::find_super_default(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// This is called during linktime when we find an invokespecial call that
|
||||||
|
// refers to a direct superinterface. It indicates that we should find the
|
||||||
|
// default method in the hierarchy of that superinterface, and if that method
|
||||||
|
// would have been a candidate from the point of view of 'this' class, then we
|
||||||
|
// return that method.
|
||||||
|
// This logic assumes that the super is a direct superclass of the caller
|
||||||
|
Method* DefaultMethods::find_super_default(
|
||||||
|
Klass* cls, Klass* super, Symbol* method_name, Symbol* sig, TRAPS) {
|
||||||
|
|
||||||
|
ResourceMark rm(THREAD);
|
||||||
|
|
||||||
|
assert(cls != NULL && super != NULL, "Need real classes");
|
||||||
|
|
||||||
|
InstanceKlass* current_class = InstanceKlass::cast(cls);
|
||||||
|
InstanceKlass* super_class = InstanceKlass::cast(super);
|
||||||
|
|
||||||
|
// Keep entire hierarchy alive for the duration of the computation
|
||||||
|
KeepAliveRegistrar keepAlive(THREAD);
|
||||||
|
KeepAliveVisitor loadKeepAlive(&keepAlive);
|
||||||
|
loadKeepAlive.run(current_class); // get hierarchy from current class
|
||||||
|
|
||||||
|
#ifndef PRODUCT
|
||||||
|
if (TraceDefaultMethods) {
|
||||||
|
tty->print_cr("Finding super default method %s.%s%s from %s",
|
||||||
|
super_class->name()->as_C_string(),
|
||||||
|
method_name->as_C_string(), sig->as_C_string(),
|
||||||
|
current_class->name()->as_C_string());
|
||||||
|
}
|
||||||
|
#endif // ndef PRODUCT
|
||||||
|
|
||||||
|
assert(super_class->is_interface(), "only call for default methods");
|
||||||
|
|
||||||
|
Method* target = NULL;
|
||||||
|
if (ParseGenericDefaults) {
|
||||||
|
target = find_generic_super_default(current_class, super_class,
|
||||||
|
method_name, sig, CHECK_NULL);
|
||||||
|
} else {
|
||||||
|
target = find_erased_super_default(current_class, super_class,
|
||||||
|
method_name, sig, CHECK_NULL);
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifndef PRODUCT
|
||||||
|
if (target != NULL) {
|
||||||
|
if (TraceDefaultMethods) {
|
||||||
|
tty->print(" Returning ");
|
||||||
|
print_method(tty, target, true);
|
||||||
|
tty->print_cr("");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif // ndef PRODUCT
|
||||||
|
return target;
|
||||||
|
}
|
||||||
|
|
||||||
|
#ifndef PRODUCT
|
||||||
|
// Return true is broad type is a covariant return of narrow type
|
||||||
|
static bool covariant_return_type(BasicType narrow, BasicType broad) {
|
||||||
|
if (narrow == broad) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (broad == T_OBJECT) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
#endif // ndef PRODUCT
|
||||||
|
|
||||||
static int assemble_redirect(
|
static int assemble_redirect(
|
||||||
BytecodeConstantPool* cp, BytecodeBuffer* buffer,
|
BytecodeConstantPool* cp, BytecodeBuffer* buffer,
|
||||||
@ -1103,7 +1356,7 @@ static int assemble_redirect(
|
|||||||
out.next();
|
out.next();
|
||||||
}
|
}
|
||||||
assert(out.at_return_type(), "Parameter counts do not match");
|
assert(out.at_return_type(), "Parameter counts do not match");
|
||||||
assert(in.type() == out.type(), "Return types are not compatible");
|
assert(covariant_return_type(out.type(), in.type()), "Return types are not compatible");
|
||||||
|
|
||||||
if (parameter_count == 1 && (in.type() == T_LONG || in.type() == T_DOUBLE)) {
|
if (parameter_count == 1 && (in.type() == T_LONG || in.type() == T_DOUBLE)) {
|
||||||
++parameter_count; // need room for return value
|
++parameter_count; // need room for return value
|
||||||
@ -1144,10 +1397,15 @@ static Method* new_method(
|
|||||||
Symbol* sig, AccessFlags flags, int max_stack, int params,
|
Symbol* sig, AccessFlags flags, int max_stack, int params,
|
||||||
ConstMethod::MethodType mt, TRAPS) {
|
ConstMethod::MethodType mt, TRAPS) {
|
||||||
|
|
||||||
address code_start = static_cast<address>(bytecodes->adr_at(0));
|
address code_start = 0;
|
||||||
int code_length = bytecodes->length();
|
int code_length = 0;
|
||||||
InlineTableSizes sizes;
|
InlineTableSizes sizes;
|
||||||
|
|
||||||
|
if (bytecodes != NULL && bytecodes->length() > 0) {
|
||||||
|
code_start = static_cast<address>(bytecodes->adr_at(0));
|
||||||
|
code_length = bytecodes->length();
|
||||||
|
}
|
||||||
|
|
||||||
Method* m = Method::allocate(cp->pool_holder()->class_loader_data(),
|
Method* m = Method::allocate(cp->pool_holder()->class_loader_data(),
|
||||||
code_length, flags, &sizes,
|
code_length, flags, &sizes,
|
||||||
mt, CHECK_NULL);
|
mt, CHECK_NULL);
|
||||||
|
@ -234,6 +234,7 @@ class java_lang_Class : AllStatic {
|
|||||||
static GrowableArray<Klass*>* _fixup_mirror_list;
|
static GrowableArray<Klass*>* _fixup_mirror_list;
|
||||||
|
|
||||||
static void set_init_lock(oop java_class, oop init_lock);
|
static void set_init_lock(oop java_class, oop init_lock);
|
||||||
|
static void set_protection_domain(oop java_class, oop protection_domain);
|
||||||
public:
|
public:
|
||||||
static void compute_offsets();
|
static void compute_offsets();
|
||||||
|
|
||||||
@ -272,7 +273,6 @@ class java_lang_Class : AllStatic {
|
|||||||
|
|
||||||
// Support for embedded per-class oops
|
// Support for embedded per-class oops
|
||||||
static oop protection_domain(oop java_class);
|
static oop protection_domain(oop java_class);
|
||||||
static void set_protection_domain(oop java_class, oop protection_domain);
|
|
||||||
static oop init_lock(oop java_class);
|
static oop init_lock(oop java_class);
|
||||||
static objArrayOop signers(oop java_class);
|
static objArrayOop signers(oop java_class);
|
||||||
static void set_signers(oop java_class, objArrayOop signers);
|
static void set_signers(oop java_class, objArrayOop signers);
|
||||||
|
@ -2017,12 +2017,6 @@ oop_since_save_marks_iterate##nv_suffix(OopClosureType* blk) { \
|
|||||||
|
|
||||||
ALL_SINCE_SAVE_MARKS_CLOSURES(CFLS_OOP_SINCE_SAVE_MARKS_DEFN)
|
ALL_SINCE_SAVE_MARKS_CLOSURES(CFLS_OOP_SINCE_SAVE_MARKS_DEFN)
|
||||||
|
|
||||||
|
|
||||||
void CompactibleFreeListSpace::object_iterate_since_last_GC(ObjectClosure* cl) {
|
|
||||||
// ugghh... how would one do this efficiently for a non-contiguous space?
|
|
||||||
guarantee(false, "NYI");
|
|
||||||
}
|
|
||||||
|
|
||||||
bool CompactibleFreeListSpace::linearAllocationWouldFail() const {
|
bool CompactibleFreeListSpace::linearAllocationWouldFail() const {
|
||||||
return _smallLinearAllocBlock._word_size == 0;
|
return _smallLinearAllocBlock._word_size == 0;
|
||||||
}
|
}
|
||||||
|
@ -396,7 +396,6 @@ class CompactibleFreeListSpace: public CompactibleSpace {
|
|||||||
// iteration support for promotion
|
// iteration support for promotion
|
||||||
void save_marks();
|
void save_marks();
|
||||||
bool no_allocs_since_save_marks();
|
bool no_allocs_since_save_marks();
|
||||||
void object_iterate_since_last_GC(ObjectClosure* cl);
|
|
||||||
|
|
||||||
// iteration support for sweeping
|
// iteration support for sweeping
|
||||||
void save_sweep_limit() {
|
void save_sweep_limit() {
|
||||||
|
@ -3129,26 +3129,6 @@ oop_since_save_marks_iterate##nv_suffix(OopClosureType* cl) { \
|
|||||||
|
|
||||||
ALL_SINCE_SAVE_MARKS_CLOSURES(CMS_SINCE_SAVE_MARKS_DEFN)
|
ALL_SINCE_SAVE_MARKS_CLOSURES(CMS_SINCE_SAVE_MARKS_DEFN)
|
||||||
|
|
||||||
void
|
|
||||||
ConcurrentMarkSweepGeneration::object_iterate_since_last_GC(ObjectClosure* blk)
|
|
||||||
{
|
|
||||||
// Not currently implemented; need to do the following. -- ysr.
|
|
||||||
// dld -- I think that is used for some sort of allocation profiler. So it
|
|
||||||
// really means the objects allocated by the mutator since the last
|
|
||||||
// GC. We could potentially implement this cheaply by recording only
|
|
||||||
// the direct allocations in a side data structure.
|
|
||||||
//
|
|
||||||
// I think we probably ought not to be required to support these
|
|
||||||
// iterations at any arbitrary point; I think there ought to be some
|
|
||||||
// call to enable/disable allocation profiling in a generation/space,
|
|
||||||
// and the iterator ought to return the objects allocated in the
|
|
||||||
// gen/space since the enable call, or the last iterator call (which
|
|
||||||
// will probably be at a GC.) That way, for gens like CM&S that would
|
|
||||||
// require some extra data structure to support this, we only pay the
|
|
||||||
// cost when it's in use...
|
|
||||||
cmsSpace()->object_iterate_since_last_GC(blk);
|
|
||||||
}
|
|
||||||
|
|
||||||
void
|
void
|
||||||
ConcurrentMarkSweepGeneration::younger_refs_iterate(OopsInGenClosure* cl) {
|
ConcurrentMarkSweepGeneration::younger_refs_iterate(OopsInGenClosure* cl) {
|
||||||
cl->set_generation(this);
|
cl->set_generation(this);
|
||||||
|
@ -1273,7 +1273,6 @@ class ConcurrentMarkSweepGeneration: public CardGeneration {
|
|||||||
// Iteration support and related enquiries
|
// Iteration support and related enquiries
|
||||||
void save_marks();
|
void save_marks();
|
||||||
bool no_allocs_since_save_marks();
|
bool no_allocs_since_save_marks();
|
||||||
void object_iterate_since_last_GC(ObjectClosure* cl);
|
|
||||||
void younger_refs_iterate(OopsInGenClosure* cl);
|
void younger_refs_iterate(OopsInGenClosure* cl);
|
||||||
|
|
||||||
// Iteration support specific to CMS generations
|
// Iteration support specific to CMS generations
|
||||||
|
@ -54,7 +54,6 @@
|
|||||||
#include "memory/referenceProcessor.hpp"
|
#include "memory/referenceProcessor.hpp"
|
||||||
#include "oops/oop.inline.hpp"
|
#include "oops/oop.inline.hpp"
|
||||||
#include "oops/oop.pcgc.inline.hpp"
|
#include "oops/oop.pcgc.inline.hpp"
|
||||||
#include "runtime/aprofiler.hpp"
|
|
||||||
#include "runtime/vmThread.hpp"
|
#include "runtime/vmThread.hpp"
|
||||||
|
|
||||||
size_t G1CollectedHeap::_humongous_object_threshold_in_words = 0;
|
size_t G1CollectedHeap::_humongous_object_threshold_in_words = 0;
|
||||||
@ -2665,11 +2664,6 @@ void G1CollectedHeap::object_iterate(ObjectClosure* cl) {
|
|||||||
heap_region_iterate(&blk);
|
heap_region_iterate(&blk);
|
||||||
}
|
}
|
||||||
|
|
||||||
void G1CollectedHeap::object_iterate_since_last_GC(ObjectClosure* cl) {
|
|
||||||
// FIXME: is this right?
|
|
||||||
guarantee(false, "object_iterate_since_last_GC not supported by G1 heap");
|
|
||||||
}
|
|
||||||
|
|
||||||
// Calls a SpaceClosure on a HeapRegion.
|
// Calls a SpaceClosure on a HeapRegion.
|
||||||
|
|
||||||
class SpaceClosureRegionClosure: public HeapRegionClosure {
|
class SpaceClosureRegionClosure: public HeapRegionClosure {
|
||||||
@ -3598,8 +3592,6 @@ G1CollectedHeap* G1CollectedHeap::heap() {
|
|||||||
void G1CollectedHeap::gc_prologue(bool full /* Ignored */) {
|
void G1CollectedHeap::gc_prologue(bool full /* Ignored */) {
|
||||||
// always_do_update_barrier = false;
|
// always_do_update_barrier = false;
|
||||||
assert(InlineCacheBuffer::is_empty(), "should have cleaned up ICBuffer");
|
assert(InlineCacheBuffer::is_empty(), "should have cleaned up ICBuffer");
|
||||||
// Call allocation profiler
|
|
||||||
AllocationProfiler::iterate_since_last_gc();
|
|
||||||
// Fill TLAB's and such
|
// Fill TLAB's and such
|
||||||
ensure_parsability(true);
|
ensure_parsability(true);
|
||||||
}
|
}
|
||||||
|
@ -1360,11 +1360,6 @@ public:
|
|||||||
object_iterate(cl);
|
object_iterate(cl);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Iterate over all objects allocated since the last collection, calling
|
|
||||||
// "cl.do_object" on each. The heap must have been initialized properly
|
|
||||||
// to support this function, or else this call will fail.
|
|
||||||
virtual void object_iterate_since_last_GC(ObjectClosure* cl);
|
|
||||||
|
|
||||||
// Iterate over all spaces in use in the heap, in ascending address order.
|
// Iterate over all spaces in use in the heap, in ascending address order.
|
||||||
virtual void space_iterate(SpaceClosure* cl);
|
virtual void space_iterate(SpaceClosure* cl);
|
||||||
|
|
||||||
|
@ -43,7 +43,6 @@
|
|||||||
#include "oops/instanceRefKlass.hpp"
|
#include "oops/instanceRefKlass.hpp"
|
||||||
#include "oops/oop.inline.hpp"
|
#include "oops/oop.inline.hpp"
|
||||||
#include "prims/jvmtiExport.hpp"
|
#include "prims/jvmtiExport.hpp"
|
||||||
#include "runtime/aprofiler.hpp"
|
|
||||||
#include "runtime/biasedLocking.hpp"
|
#include "runtime/biasedLocking.hpp"
|
||||||
#include "runtime/fprofiler.hpp"
|
#include "runtime/fprofiler.hpp"
|
||||||
#include "runtime/synchronizer.hpp"
|
#include "runtime/synchronizer.hpp"
|
||||||
|
@ -23,12 +23,14 @@
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
#include "precompiled.hpp"
|
#include "precompiled.hpp"
|
||||||
|
#include "gc_implementation/shared/copyFailedInfo.hpp"
|
||||||
#include "gc_implementation/shared/gcHeapSummary.hpp"
|
#include "gc_implementation/shared/gcHeapSummary.hpp"
|
||||||
#include "gc_implementation/shared/gcTimer.hpp"
|
#include "gc_implementation/shared/gcTimer.hpp"
|
||||||
#include "gc_implementation/shared/gcTrace.hpp"
|
#include "gc_implementation/shared/gcTrace.hpp"
|
||||||
#include "gc_implementation/shared/copyFailedInfo.hpp"
|
#include "gc_implementation/shared/objectCountEventSender.hpp"
|
||||||
#include "memory/heapInspection.hpp"
|
#include "memory/heapInspection.hpp"
|
||||||
#include "memory/referenceProcessorStats.hpp"
|
#include "memory/referenceProcessorStats.hpp"
|
||||||
|
#include "runtime/os.hpp"
|
||||||
#include "utilities/globalDefinitions.hpp"
|
#include "utilities/globalDefinitions.hpp"
|
||||||
|
|
||||||
#if INCLUDE_ALL_GCS
|
#if INCLUDE_ALL_GCS
|
||||||
@ -38,7 +40,7 @@
|
|||||||
#define assert_unset_gc_id() assert(_shared_gc_info.id() == SharedGCInfo::UNSET_GCID, "GC already started?")
|
#define assert_unset_gc_id() assert(_shared_gc_info.id() == SharedGCInfo::UNSET_GCID, "GC already started?")
|
||||||
#define assert_set_gc_id() assert(_shared_gc_info.id() != SharedGCInfo::UNSET_GCID, "GC not started?")
|
#define assert_set_gc_id() assert(_shared_gc_info.id() != SharedGCInfo::UNSET_GCID, "GC not started?")
|
||||||
|
|
||||||
static jlong GCTracer_next_gc_id = 0;
|
static GCId GCTracer_next_gc_id = 0;
|
||||||
static GCId create_new_gc_id() {
|
static GCId create_new_gc_id() {
|
||||||
return GCTracer_next_gc_id++;
|
return GCTracer_next_gc_id++;
|
||||||
}
|
}
|
||||||
@ -91,26 +93,38 @@ void GCTracer::report_gc_reference_stats(const ReferenceProcessorStats& rps) con
|
|||||||
}
|
}
|
||||||
|
|
||||||
#if INCLUDE_SERVICES
|
#if INCLUDE_SERVICES
|
||||||
void ObjectCountEventSenderClosure::do_cinfo(KlassInfoEntry* entry) {
|
class ObjectCountEventSenderClosure : public KlassInfoClosure {
|
||||||
|
const GCId _gc_id;
|
||||||
|
const double _size_threshold_percentage;
|
||||||
|
const size_t _total_size_in_words;
|
||||||
|
const jlong _timestamp;
|
||||||
|
|
||||||
|
public:
|
||||||
|
ObjectCountEventSenderClosure(GCId gc_id, size_t total_size_in_words, jlong timestamp) :
|
||||||
|
_gc_id(gc_id),
|
||||||
|
_size_threshold_percentage(ObjectCountCutOffPercent / 100),
|
||||||
|
_total_size_in_words(total_size_in_words),
|
||||||
|
_timestamp(timestamp)
|
||||||
|
{}
|
||||||
|
|
||||||
|
virtual void do_cinfo(KlassInfoEntry* entry) {
|
||||||
if (should_send_event(entry)) {
|
if (should_send_event(entry)) {
|
||||||
send_event(entry);
|
ObjectCountEventSender::send(entry, _gc_id, _timestamp);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
void ObjectCountEventSenderClosure::send_event(KlassInfoEntry* entry) {
|
private:
|
||||||
_gc_tracer->send_object_count_after_gc_event(entry->klass(), entry->count(),
|
bool should_send_event(const KlassInfoEntry* entry) const {
|
||||||
entry->words() * BytesPerWord);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool ObjectCountEventSenderClosure::should_send_event(KlassInfoEntry* entry) const {
|
|
||||||
double percentage_of_heap = ((double) entry->words()) / _total_size_in_words;
|
double percentage_of_heap = ((double) entry->words()) / _total_size_in_words;
|
||||||
return percentage_of_heap > _size_threshold_percentage;
|
return percentage_of_heap >= _size_threshold_percentage;
|
||||||
}
|
}
|
||||||
|
};
|
||||||
|
|
||||||
void GCTracer::report_object_count_after_gc(BoolObjectClosure* is_alive_cl) {
|
void GCTracer::report_object_count_after_gc(BoolObjectClosure* is_alive_cl) {
|
||||||
assert_set_gc_id();
|
assert_set_gc_id();
|
||||||
|
assert(is_alive_cl != NULL, "Must supply function to check liveness");
|
||||||
|
|
||||||
if (should_send_object_count_after_gc_event()) {
|
if (ObjectCountEventSender::should_send_event()) {
|
||||||
ResourceMark rm;
|
ResourceMark rm;
|
||||||
|
|
||||||
KlassInfoTable cit(false);
|
KlassInfoTable cit(false);
|
||||||
@ -118,12 +132,13 @@ void GCTracer::report_object_count_after_gc(BoolObjectClosure* is_alive_cl) {
|
|||||||
HeapInspection hi(false, false, false, NULL);
|
HeapInspection hi(false, false, false, NULL);
|
||||||
hi.populate_table(&cit, is_alive_cl);
|
hi.populate_table(&cit, is_alive_cl);
|
||||||
|
|
||||||
ObjectCountEventSenderClosure event_sender(this, cit.size_of_instances_in_words());
|
jlong timestamp = os::elapsed_counter();
|
||||||
|
ObjectCountEventSenderClosure event_sender(_shared_gc_info.id(), cit.size_of_instances_in_words(), timestamp);
|
||||||
cit.iterate(&event_sender);
|
cit.iterate(&event_sender);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
#endif
|
#endif // INCLUDE_SERVICES
|
||||||
|
|
||||||
void GCTracer::report_gc_heap_summary(GCWhen::Type when, const GCHeapSummary& heap_summary, const MetaspaceSummary& meta_space_summary) const {
|
void GCTracer::report_gc_heap_summary(GCWhen::Type when, const GCHeapSummary& heap_summary, const MetaspaceSummary& meta_space_summary) const {
|
||||||
assert_set_gc_id();
|
assert_set_gc_id();
|
||||||
|
@ -30,7 +30,6 @@
|
|||||||
#include "gc_implementation/shared/gcWhen.hpp"
|
#include "gc_implementation/shared/gcWhen.hpp"
|
||||||
#include "gc_implementation/shared/copyFailedInfo.hpp"
|
#include "gc_implementation/shared/copyFailedInfo.hpp"
|
||||||
#include "memory/allocation.hpp"
|
#include "memory/allocation.hpp"
|
||||||
#include "memory/klassInfoClosure.hpp"
|
|
||||||
#include "memory/referenceType.hpp"
|
#include "memory/referenceType.hpp"
|
||||||
#if INCLUDE_ALL_GCS
|
#if INCLUDE_ALL_GCS
|
||||||
#include "gc_implementation/g1/g1YCTypes.hpp"
|
#include "gc_implementation/g1/g1YCTypes.hpp"
|
||||||
@ -113,7 +112,6 @@ class G1YoungGCInfo VALUE_OBJ_CLASS_SPEC {
|
|||||||
#endif // INCLUDE_ALL_GCS
|
#endif // INCLUDE_ALL_GCS
|
||||||
|
|
||||||
class GCTracer : public ResourceObj {
|
class GCTracer : public ResourceObj {
|
||||||
friend class ObjectCountEventSenderClosure;
|
|
||||||
protected:
|
protected:
|
||||||
SharedGCInfo _shared_gc_info;
|
SharedGCInfo _shared_gc_info;
|
||||||
|
|
||||||
@ -123,7 +121,6 @@ class GCTracer : public ResourceObj {
|
|||||||
void report_gc_heap_summary(GCWhen::Type when, const GCHeapSummary& heap_summary, const MetaspaceSummary& meta_space_summary) const;
|
void report_gc_heap_summary(GCWhen::Type when, const GCHeapSummary& heap_summary, const MetaspaceSummary& meta_space_summary) const;
|
||||||
void report_gc_reference_stats(const ReferenceProcessorStats& rp) const;
|
void report_gc_reference_stats(const ReferenceProcessorStats& rp) const;
|
||||||
void report_object_count_after_gc(BoolObjectClosure* object_filter) NOT_SERVICES_RETURN;
|
void report_object_count_after_gc(BoolObjectClosure* object_filter) NOT_SERVICES_RETURN;
|
||||||
|
|
||||||
bool has_reported_gc_start() const;
|
bool has_reported_gc_start() const;
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
@ -137,25 +134,6 @@ class GCTracer : public ResourceObj {
|
|||||||
void send_meta_space_summary_event(GCWhen::Type when, const MetaspaceSummary& meta_space_summary) const;
|
void send_meta_space_summary_event(GCWhen::Type when, const MetaspaceSummary& meta_space_summary) const;
|
||||||
void send_reference_stats_event(ReferenceType type, size_t count) const;
|
void send_reference_stats_event(ReferenceType type, size_t count) const;
|
||||||
void send_phase_events(TimePartitions* time_partitions) const;
|
void send_phase_events(TimePartitions* time_partitions) const;
|
||||||
void send_object_count_after_gc_event(Klass* klass, jlong count, julong total_size) const NOT_SERVICES_RETURN;
|
|
||||||
bool should_send_object_count_after_gc_event() const;
|
|
||||||
};
|
|
||||||
|
|
||||||
class ObjectCountEventSenderClosure : public KlassInfoClosure {
|
|
||||||
GCTracer* _gc_tracer;
|
|
||||||
const double _size_threshold_percentage;
|
|
||||||
const size_t _total_size_in_words;
|
|
||||||
public:
|
|
||||||
ObjectCountEventSenderClosure(GCTracer* gc_tracer, size_t total_size_in_words) :
|
|
||||||
_gc_tracer(gc_tracer),
|
|
||||||
_size_threshold_percentage(ObjectCountCutOffPercent / 100),
|
|
||||||
_total_size_in_words(total_size_in_words)
|
|
||||||
{}
|
|
||||||
virtual void do_cinfo(KlassInfoEntry* entry);
|
|
||||||
protected:
|
|
||||||
virtual void send_event(KlassInfoEntry* entry);
|
|
||||||
private:
|
|
||||||
bool should_send_event(KlassInfoEntry* entry) const;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
class YoungGCTracer : public GCTracer {
|
class YoungGCTracer : public GCTracer {
|
||||||
|
@ -123,27 +123,6 @@ void OldGCTracer::send_concurrent_mode_failure_event() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#if INCLUDE_SERVICES
|
|
||||||
void GCTracer::send_object_count_after_gc_event(Klass* klass, jlong count, julong total_size) const {
|
|
||||||
EventObjectCountAfterGC e;
|
|
||||||
if (e.should_commit()) {
|
|
||||||
e.set_gcId(_shared_gc_info.id());
|
|
||||||
e.set_class(klass);
|
|
||||||
e.set_count(count);
|
|
||||||
e.set_totalSize(total_size);
|
|
||||||
e.commit();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
bool GCTracer::should_send_object_count_after_gc_event() const {
|
|
||||||
#if INCLUDE_TRACE
|
|
||||||
return Tracing::is_event_enabled(EventObjectCountAfterGC::eventId);
|
|
||||||
#else
|
|
||||||
return false;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
#if INCLUDE_ALL_GCS
|
#if INCLUDE_ALL_GCS
|
||||||
void G1NewTracer::send_g1_young_gc_event() {
|
void G1NewTracer::send_g1_young_gc_event() {
|
||||||
EventGCG1GarbageCollection e(UNTIMED);
|
EventGCG1GarbageCollection e(UNTIMED);
|
||||||
|
@ -0,0 +1,55 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2013, Oracle and/or its affiliates. All rights reserved.
|
||||||
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
|
*
|
||||||
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU General Public License version 2 only, as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||||||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||||
|
* version 2 for more details (a copy is included in the LICENSE file that
|
||||||
|
* accompanied this code).
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License version
|
||||||
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
*
|
||||||
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||||||
|
* or visit www.oracle.com if you need additional information or have any
|
||||||
|
* questions.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
#include "precompiled.hpp"
|
||||||
|
#include "gc_implementation/shared/objectCountEventSender.hpp"
|
||||||
|
#include "memory/heapInspection.hpp"
|
||||||
|
#include "trace/tracing.hpp"
|
||||||
|
#include "utilities/globalDefinitions.hpp"
|
||||||
|
|
||||||
|
#if INCLUDE_SERVICES
|
||||||
|
|
||||||
|
void ObjectCountEventSender::send(const KlassInfoEntry* entry, GCId gc_id, jlong timestamp) {
|
||||||
|
assert(Tracing::is_event_enabled(EventObjectCountAfterGC::eventId),
|
||||||
|
"Only call this method if the event is enabled");
|
||||||
|
|
||||||
|
EventObjectCountAfterGC event(UNTIMED);
|
||||||
|
event.set_gcId(gc_id);
|
||||||
|
event.set_class(entry->klass());
|
||||||
|
event.set_count(entry->count());
|
||||||
|
event.set_totalSize(entry->words() * BytesPerWord);
|
||||||
|
event.set_endtime(timestamp);
|
||||||
|
event.commit();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool ObjectCountEventSender::should_send_event() {
|
||||||
|
#if INCLUDE_TRACE
|
||||||
|
return Tracing::is_event_enabled(EventObjectCountAfterGC::eventId);
|
||||||
|
#else
|
||||||
|
return false;
|
||||||
|
#endif // INCLUDE_TRACE
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif // INCLUDE_SERVICES
|
@ -22,15 +22,23 @@
|
|||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
|
|
||||||
#ifndef SHARE_VM_MEMORY_KLASSINFOCLOSURE_HPP
|
#ifndef SHARE_VM_OBJECT_COUNT_EVENT_SENDER_HPP
|
||||||
#define SHARE_VM_MEMORY_KLASSINFOCLOSURE_HPP
|
#define SHARE_VM_OBJECT_COUNT_EVENT_SENDER_HPP
|
||||||
|
|
||||||
|
#include "gc_implementation/shared/gcTrace.hpp"
|
||||||
|
#include "memory/allocation.hpp"
|
||||||
|
#include "utilities/macros.hpp"
|
||||||
|
|
||||||
|
#if INCLUDE_SERVICES
|
||||||
|
|
||||||
class KlassInfoEntry;
|
class KlassInfoEntry;
|
||||||
|
|
||||||
class KlassInfoClosure : public StackObj {
|
class ObjectCountEventSender : public AllStatic {
|
||||||
public:
|
public:
|
||||||
// Called for each KlassInfoEntry.
|
static void send(const KlassInfoEntry* entry, GCId gc_id, jlong timestamp);
|
||||||
virtual void do_cinfo(KlassInfoEntry* cie) = 0;
|
static bool should_send_event();
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif // SHARE_VM_MEMORY_KLASSINFOCLOSURE_HPP
|
#endif // INCLUDE_SERVICES
|
||||||
|
|
||||||
|
#endif // SHARE_VM_OBJECT_COUNT_EVENT_SENDER
|
@ -85,16 +85,16 @@ GCHeapSummary CollectedHeap::create_heap_summary() {
|
|||||||
|
|
||||||
MetaspaceSummary CollectedHeap::create_metaspace_summary() {
|
MetaspaceSummary CollectedHeap::create_metaspace_summary() {
|
||||||
const MetaspaceSizes meta_space(
|
const MetaspaceSizes meta_space(
|
||||||
0, /*MetaspaceAux::capacity_in_bytes(),*/
|
MetaspaceAux::allocated_capacity_bytes(),
|
||||||
0, /*MetaspaceAux::used_in_bytes(),*/
|
MetaspaceAux::allocated_used_bytes(),
|
||||||
MetaspaceAux::reserved_in_bytes());
|
MetaspaceAux::reserved_in_bytes());
|
||||||
const MetaspaceSizes data_space(
|
const MetaspaceSizes data_space(
|
||||||
0, /*MetaspaceAux::capacity_in_bytes(Metaspace::NonClassType),*/
|
MetaspaceAux::allocated_capacity_bytes(Metaspace::NonClassType),
|
||||||
0, /*MetaspaceAux::used_in_bytes(Metaspace::NonClassType),*/
|
MetaspaceAux::allocated_used_bytes(Metaspace::NonClassType),
|
||||||
MetaspaceAux::reserved_in_bytes(Metaspace::NonClassType));
|
MetaspaceAux::reserved_in_bytes(Metaspace::NonClassType));
|
||||||
const MetaspaceSizes class_space(
|
const MetaspaceSizes class_space(
|
||||||
0, /*MetaspaceAux::capacity_in_bytes(Metaspace::ClassType),*/
|
MetaspaceAux::allocated_capacity_bytes(Metaspace::ClassType),
|
||||||
0, /*MetaspaceAux::used_in_bytes(Metaspace::ClassType),*/
|
MetaspaceAux::allocated_used_bytes(Metaspace::ClassType),
|
||||||
MetaspaceAux::reserved_in_bytes(Metaspace::ClassType));
|
MetaspaceAux::reserved_in_bytes(Metaspace::ClassType));
|
||||||
|
|
||||||
return MetaspaceSummary(meta_space, data_space, class_space);
|
return MetaspaceSummary(meta_space, data_space, class_space);
|
||||||
|
@ -236,10 +236,11 @@ class ChunkPool: public CHeapObj<mtInternal> {
|
|||||||
size_t _num_used; // number of chunks currently checked out
|
size_t _num_used; // number of chunks currently checked out
|
||||||
const size_t _size; // size of each chunk (must be uniform)
|
const size_t _size; // size of each chunk (must be uniform)
|
||||||
|
|
||||||
// Our three static pools
|
// Our four static pools
|
||||||
static ChunkPool* _large_pool;
|
static ChunkPool* _large_pool;
|
||||||
static ChunkPool* _medium_pool;
|
static ChunkPool* _medium_pool;
|
||||||
static ChunkPool* _small_pool;
|
static ChunkPool* _small_pool;
|
||||||
|
static ChunkPool* _tiny_pool;
|
||||||
|
|
||||||
// return first element or null
|
// return first element or null
|
||||||
void* get_first() {
|
void* get_first() {
|
||||||
@ -319,15 +320,18 @@ class ChunkPool: public CHeapObj<mtInternal> {
|
|||||||
static ChunkPool* large_pool() { assert(_large_pool != NULL, "must be initialized"); return _large_pool; }
|
static ChunkPool* large_pool() { assert(_large_pool != NULL, "must be initialized"); return _large_pool; }
|
||||||
static ChunkPool* medium_pool() { assert(_medium_pool != NULL, "must be initialized"); return _medium_pool; }
|
static ChunkPool* medium_pool() { assert(_medium_pool != NULL, "must be initialized"); return _medium_pool; }
|
||||||
static ChunkPool* small_pool() { assert(_small_pool != NULL, "must be initialized"); return _small_pool; }
|
static ChunkPool* small_pool() { assert(_small_pool != NULL, "must be initialized"); return _small_pool; }
|
||||||
|
static ChunkPool* tiny_pool() { assert(_tiny_pool != NULL, "must be initialized"); return _tiny_pool; }
|
||||||
|
|
||||||
static void initialize() {
|
static void initialize() {
|
||||||
_large_pool = new ChunkPool(Chunk::size + Chunk::aligned_overhead_size());
|
_large_pool = new ChunkPool(Chunk::size + Chunk::aligned_overhead_size());
|
||||||
_medium_pool = new ChunkPool(Chunk::medium_size + Chunk::aligned_overhead_size());
|
_medium_pool = new ChunkPool(Chunk::medium_size + Chunk::aligned_overhead_size());
|
||||||
_small_pool = new ChunkPool(Chunk::init_size + Chunk::aligned_overhead_size());
|
_small_pool = new ChunkPool(Chunk::init_size + Chunk::aligned_overhead_size());
|
||||||
|
_tiny_pool = new ChunkPool(Chunk::tiny_size + Chunk::aligned_overhead_size());
|
||||||
}
|
}
|
||||||
|
|
||||||
static void clean() {
|
static void clean() {
|
||||||
enum { BlocksToKeep = 5 };
|
enum { BlocksToKeep = 5 };
|
||||||
|
_tiny_pool->free_all_but(BlocksToKeep);
|
||||||
_small_pool->free_all_but(BlocksToKeep);
|
_small_pool->free_all_but(BlocksToKeep);
|
||||||
_medium_pool->free_all_but(BlocksToKeep);
|
_medium_pool->free_all_but(BlocksToKeep);
|
||||||
_large_pool->free_all_but(BlocksToKeep);
|
_large_pool->free_all_but(BlocksToKeep);
|
||||||
@ -337,6 +341,7 @@ class ChunkPool: public CHeapObj<mtInternal> {
|
|||||||
ChunkPool* ChunkPool::_large_pool = NULL;
|
ChunkPool* ChunkPool::_large_pool = NULL;
|
||||||
ChunkPool* ChunkPool::_medium_pool = NULL;
|
ChunkPool* ChunkPool::_medium_pool = NULL;
|
||||||
ChunkPool* ChunkPool::_small_pool = NULL;
|
ChunkPool* ChunkPool::_small_pool = NULL;
|
||||||
|
ChunkPool* ChunkPool::_tiny_pool = NULL;
|
||||||
|
|
||||||
void chunkpool_init() {
|
void chunkpool_init() {
|
||||||
ChunkPool::initialize();
|
ChunkPool::initialize();
|
||||||
@ -376,6 +381,7 @@ void* Chunk::operator new (size_t requested_size, AllocFailType alloc_failmode,
|
|||||||
case Chunk::size: return ChunkPool::large_pool()->allocate(bytes, alloc_failmode);
|
case Chunk::size: return ChunkPool::large_pool()->allocate(bytes, alloc_failmode);
|
||||||
case Chunk::medium_size: return ChunkPool::medium_pool()->allocate(bytes, alloc_failmode);
|
case Chunk::medium_size: return ChunkPool::medium_pool()->allocate(bytes, alloc_failmode);
|
||||||
case Chunk::init_size: return ChunkPool::small_pool()->allocate(bytes, alloc_failmode);
|
case Chunk::init_size: return ChunkPool::small_pool()->allocate(bytes, alloc_failmode);
|
||||||
|
case Chunk::tiny_size: return ChunkPool::tiny_pool()->allocate(bytes, alloc_failmode);
|
||||||
default: {
|
default: {
|
||||||
void* p = os::malloc(bytes, mtChunk, CALLER_PC);
|
void* p = os::malloc(bytes, mtChunk, CALLER_PC);
|
||||||
if (p == NULL && alloc_failmode == AllocFailStrategy::EXIT_OOM) {
|
if (p == NULL && alloc_failmode == AllocFailStrategy::EXIT_OOM) {
|
||||||
@ -392,6 +398,7 @@ void Chunk::operator delete(void* p) {
|
|||||||
case Chunk::size: ChunkPool::large_pool()->free(c); break;
|
case Chunk::size: ChunkPool::large_pool()->free(c); break;
|
||||||
case Chunk::medium_size: ChunkPool::medium_pool()->free(c); break;
|
case Chunk::medium_size: ChunkPool::medium_pool()->free(c); break;
|
||||||
case Chunk::init_size: ChunkPool::small_pool()->free(c); break;
|
case Chunk::init_size: ChunkPool::small_pool()->free(c); break;
|
||||||
|
case Chunk::tiny_size: ChunkPool::tiny_pool()->free(c); break;
|
||||||
default: os::free(c, mtChunk);
|
default: os::free(c, mtChunk);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -353,7 +353,8 @@ class Chunk: CHeapObj<mtChunk> {
|
|||||||
slack = 20, // suspected sizeof(Chunk) + internal malloc headers
|
slack = 20, // suspected sizeof(Chunk) + internal malloc headers
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
init_size = 1*K - slack, // Size of first chunk
|
tiny_size = 256 - slack, // Size of first chunk (tiny)
|
||||||
|
init_size = 1*K - slack, // Size of first chunk (normal aka small)
|
||||||
medium_size= 10*K - slack, // Size of medium-sized chunk
|
medium_size= 10*K - slack, // Size of medium-sized chunk
|
||||||
size = 32*K - slack, // Default size of an Arena chunk (following the first)
|
size = 32*K - slack, // Default size of an Arena chunk (following the first)
|
||||||
non_pool_size = init_size + 32 // An initial size which is not one of above
|
non_pool_size = init_size + 32 // An initial size which is not one of above
|
||||||
|
@ -450,11 +450,6 @@ void DefNewGeneration::compute_new_size() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void DefNewGeneration::object_iterate_since_last_GC(ObjectClosure* cl) {
|
|
||||||
// $$$ This may be wrong in case of "scavenge failure"?
|
|
||||||
eden()->object_iterate(cl);
|
|
||||||
}
|
|
||||||
|
|
||||||
void DefNewGeneration::younger_refs_iterate(OopsInGenClosure* cl) {
|
void DefNewGeneration::younger_refs_iterate(OopsInGenClosure* cl) {
|
||||||
assert(false, "NYI -- are you sure you want to call this?");
|
assert(false, "NYI -- are you sure you want to call this?");
|
||||||
}
|
}
|
||||||
|
@ -252,7 +252,6 @@ protected:
|
|||||||
|
|
||||||
// Iteration
|
// Iteration
|
||||||
void object_iterate(ObjectClosure* blk);
|
void object_iterate(ObjectClosure* blk);
|
||||||
void object_iterate_since_last_GC(ObjectClosure* cl);
|
|
||||||
|
|
||||||
void younger_refs_iterate(OopsInGenClosure* cl);
|
void younger_refs_iterate(OopsInGenClosure* cl);
|
||||||
|
|
||||||
|
@ -42,7 +42,6 @@
|
|||||||
#include "memory/space.hpp"
|
#include "memory/space.hpp"
|
||||||
#include "oops/oop.inline.hpp"
|
#include "oops/oop.inline.hpp"
|
||||||
#include "oops/oop.inline2.hpp"
|
#include "oops/oop.inline2.hpp"
|
||||||
#include "runtime/aprofiler.hpp"
|
|
||||||
#include "runtime/biasedLocking.hpp"
|
#include "runtime/biasedLocking.hpp"
|
||||||
#include "runtime/fprofiler.hpp"
|
#include "runtime/fprofiler.hpp"
|
||||||
#include "runtime/handles.hpp"
|
#include "runtime/handles.hpp"
|
||||||
@ -873,12 +872,6 @@ void GenCollectedHeap::safe_object_iterate(ObjectClosure* cl) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void GenCollectedHeap::object_iterate_since_last_GC(ObjectClosure* cl) {
|
|
||||||
for (int i = 0; i < _n_gens; i++) {
|
|
||||||
_gens[i]->object_iterate_since_last_GC(cl);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Space* GenCollectedHeap::space_containing(const void* addr) const {
|
Space* GenCollectedHeap::space_containing(const void* addr) const {
|
||||||
for (int i = 0; i < _n_gens; i++) {
|
for (int i = 0; i < _n_gens; i++) {
|
||||||
Space* res = _gens[i]->space_containing(addr);
|
Space* res = _gens[i]->space_containing(addr);
|
||||||
@ -1186,8 +1179,6 @@ void GenCollectedHeap::gc_prologue(bool full) {
|
|||||||
CollectedHeap::accumulate_statistics_all_tlabs();
|
CollectedHeap::accumulate_statistics_all_tlabs();
|
||||||
ensure_parsability(true); // retire TLABs
|
ensure_parsability(true); // retire TLABs
|
||||||
|
|
||||||
// Call allocation profiler
|
|
||||||
AllocationProfiler::iterate_since_last_gc();
|
|
||||||
// Walk generations
|
// Walk generations
|
||||||
GenGCPrologueClosure blk(full);
|
GenGCPrologueClosure blk(full);
|
||||||
generation_iterate(&blk, false); // not old-to-young.
|
generation_iterate(&blk, false); // not old-to-young.
|
||||||
|
@ -222,7 +222,6 @@ public:
|
|||||||
void oop_iterate(MemRegion mr, ExtendedOopClosure* cl);
|
void oop_iterate(MemRegion mr, ExtendedOopClosure* cl);
|
||||||
void object_iterate(ObjectClosure* cl);
|
void object_iterate(ObjectClosure* cl);
|
||||||
void safe_object_iterate(ObjectClosure* cl);
|
void safe_object_iterate(ObjectClosure* cl);
|
||||||
void object_iterate_since_last_GC(ObjectClosure* cl);
|
|
||||||
Space* space_containing(const void* addr) const;
|
Space* space_containing(const void* addr) const;
|
||||||
|
|
||||||
// A CollectedHeap is divided into a dense sequence of "blocks"; that is,
|
// A CollectedHeap is divided into a dense sequence of "blocks"; that is,
|
||||||
|
@ -811,16 +811,6 @@ void OneContigSpaceCardGeneration::space_iterate(SpaceClosure* blk,
|
|||||||
blk->do_space(_the_space);
|
blk->do_space(_the_space);
|
||||||
}
|
}
|
||||||
|
|
||||||
void OneContigSpaceCardGeneration::object_iterate_since_last_GC(ObjectClosure* blk) {
|
|
||||||
// Deal with delayed initialization of _the_space,
|
|
||||||
// and lack of initialization of _last_gc.
|
|
||||||
if (_last_gc.space() == NULL) {
|
|
||||||
assert(the_space() != NULL, "shouldn't be NULL");
|
|
||||||
_last_gc = the_space()->bottom_mark();
|
|
||||||
}
|
|
||||||
the_space()->object_iterate_from(_last_gc, blk);
|
|
||||||
}
|
|
||||||
|
|
||||||
void OneContigSpaceCardGeneration::younger_refs_iterate(OopsInGenClosure* blk) {
|
void OneContigSpaceCardGeneration::younger_refs_iterate(OopsInGenClosure* blk) {
|
||||||
blk->set_generation(this);
|
blk->set_generation(this);
|
||||||
younger_refs_in_space_iterate(_the_space, blk);
|
younger_refs_in_space_iterate(_the_space, blk);
|
||||||
|
@ -551,12 +551,6 @@ class Generation: public CHeapObj<mtGC> {
|
|||||||
// the heap. This defaults to object_iterate() unless overridden.
|
// the heap. This defaults to object_iterate() unless overridden.
|
||||||
virtual void safe_object_iterate(ObjectClosure* cl);
|
virtual void safe_object_iterate(ObjectClosure* cl);
|
||||||
|
|
||||||
// Iterate over all objects allocated in the generation since the last
|
|
||||||
// collection, calling "cl.do_object" on each. The generation must have
|
|
||||||
// been initialized properly to support this function, or else this call
|
|
||||||
// will fail.
|
|
||||||
virtual void object_iterate_since_last_GC(ObjectClosure* cl) = 0;
|
|
||||||
|
|
||||||
// Apply "cl->do_oop" to (the address of) all and only all the ref fields
|
// Apply "cl->do_oop" to (the address of) all and only all the ref fields
|
||||||
// in the current generation that contain pointers to objects in younger
|
// in the current generation that contain pointers to objects in younger
|
||||||
// generations. Objects allocated since the last "save_marks" call are
|
// generations. Objects allocated since the last "save_marks" call are
|
||||||
@ -724,7 +718,6 @@ class OneContigSpaceCardGeneration: public CardGeneration {
|
|||||||
// Iteration
|
// Iteration
|
||||||
void object_iterate(ObjectClosure* blk);
|
void object_iterate(ObjectClosure* blk);
|
||||||
void space_iterate(SpaceClosure* blk, bool usedOnly = false);
|
void space_iterate(SpaceClosure* blk, bool usedOnly = false);
|
||||||
void object_iterate_since_last_GC(ObjectClosure* cl);
|
|
||||||
|
|
||||||
void younger_refs_iterate(OopsInGenClosure* blk);
|
void younger_refs_iterate(OopsInGenClosure* blk);
|
||||||
|
|
||||||
|
@ -26,7 +26,6 @@
|
|||||||
#define SHARE_VM_MEMORY_HEAPINSPECTION_HPP
|
#define SHARE_VM_MEMORY_HEAPINSPECTION_HPP
|
||||||
|
|
||||||
#include "memory/allocation.inline.hpp"
|
#include "memory/allocation.inline.hpp"
|
||||||
#include "memory/klassInfoClosure.hpp"
|
|
||||||
#include "oops/oop.inline.hpp"
|
#include "oops/oop.inline.hpp"
|
||||||
#include "oops/annotations.hpp"
|
#include "oops/annotations.hpp"
|
||||||
#include "utilities/macros.hpp"
|
#include "utilities/macros.hpp"
|
||||||
@ -204,6 +203,12 @@ class KlassInfoEntry: public CHeapObj<mtInternal> {
|
|||||||
const char* name() const;
|
const char* name() const;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
class KlassInfoClosure : public StackObj {
|
||||||
|
public:
|
||||||
|
// Called for each KlassInfoEntry.
|
||||||
|
virtual void do_cinfo(KlassInfoEntry* cie) = 0;
|
||||||
|
};
|
||||||
|
|
||||||
class KlassInfoBucket: public CHeapObj<mtInternal> {
|
class KlassInfoBucket: public CHeapObj<mtInternal> {
|
||||||
private:
|
private:
|
||||||
KlassInfoEntry* _list;
|
KlassInfoEntry* _list;
|
||||||
|
@ -166,11 +166,6 @@ public:
|
|||||||
// Same as above, restricted to a memory region.
|
// Same as above, restricted to a memory region.
|
||||||
virtual void oop_iterate(MemRegion mr, ExtendedOopClosure* cl) = 0;
|
virtual void oop_iterate(MemRegion mr, ExtendedOopClosure* cl) = 0;
|
||||||
|
|
||||||
// Iterate over all objects allocated since the last collection, calling
|
|
||||||
// "cl->do_object" on each. The heap must have been initialized properly
|
|
||||||
// to support this function, or else this call will fail.
|
|
||||||
virtual void object_iterate_since_last_GC(ObjectClosure* cl) = 0;
|
|
||||||
|
|
||||||
// Iterate over all spaces in use in the heap, in an undefined order.
|
// Iterate over all spaces in use in the heap, in an undefined order.
|
||||||
virtual void space_iterate(SpaceClosure* cl) = 0;
|
virtual void space_iterate(SpaceClosure* cl) = 0;
|
||||||
|
|
||||||
|
@ -52,7 +52,6 @@
|
|||||||
#include "oops/oop.inline.hpp"
|
#include "oops/oop.inline.hpp"
|
||||||
#include "oops/typeArrayKlass.hpp"
|
#include "oops/typeArrayKlass.hpp"
|
||||||
#include "prims/jvmtiRedefineClassesTrace.hpp"
|
#include "prims/jvmtiRedefineClassesTrace.hpp"
|
||||||
#include "runtime/aprofiler.hpp"
|
|
||||||
#include "runtime/arguments.hpp"
|
#include "runtime/arguments.hpp"
|
||||||
#include "runtime/deoptimization.hpp"
|
#include "runtime/deoptimization.hpp"
|
||||||
#include "runtime/fprofiler.hpp"
|
#include "runtime/fprofiler.hpp"
|
||||||
|
@ -71,7 +71,6 @@ Method* ArrayKlass::uncached_lookup_method(Symbol* name, Symbol* signature) cons
|
|||||||
}
|
}
|
||||||
|
|
||||||
ArrayKlass::ArrayKlass(Symbol* name) {
|
ArrayKlass::ArrayKlass(Symbol* name) {
|
||||||
set_alloc_size(0);
|
|
||||||
set_name(name);
|
set_name(name);
|
||||||
|
|
||||||
set_super(Universe::is_bootstrapping() ? (Klass*)NULL : SystemDictionary::Object_klass());
|
set_super(Universe::is_bootstrapping() ? (Klass*)NULL : SystemDictionary::Object_klass());
|
||||||
@ -161,12 +160,6 @@ void ArrayKlass::array_klasses_do(void f(Klass* k)) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void ArrayKlass::with_array_klasses_do(void f(Klass* k)) {
|
|
||||||
array_klasses_do(f);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// GC support
|
// GC support
|
||||||
|
|
||||||
void ArrayKlass::oops_do(OopClosure* cl) {
|
void ArrayKlass::oops_do(OopClosure* cl) {
|
||||||
|
@ -39,7 +39,6 @@ class ArrayKlass: public Klass {
|
|||||||
Klass* volatile _higher_dimension; // Refers the (n+1)'th-dimensional array (if present).
|
Klass* volatile _higher_dimension; // Refers the (n+1)'th-dimensional array (if present).
|
||||||
Klass* volatile _lower_dimension; // Refers the (n-1)'th-dimensional array (if present).
|
Klass* volatile _lower_dimension; // Refers the (n-1)'th-dimensional array (if present).
|
||||||
int _vtable_len; // size of vtable for this klass
|
int _vtable_len; // size of vtable for this klass
|
||||||
juint _alloc_size; // allocation profiling support
|
|
||||||
oop _component_mirror; // component type, as a java/lang/Class
|
oop _component_mirror; // component type, as a java/lang/Class
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
@ -65,10 +64,6 @@ class ArrayKlass: public Klass {
|
|||||||
void set_lower_dimension(Klass* k) { _lower_dimension = k; }
|
void set_lower_dimension(Klass* k) { _lower_dimension = k; }
|
||||||
Klass** adr_lower_dimension() { return (Klass**)&this->_lower_dimension;}
|
Klass** adr_lower_dimension() { return (Klass**)&this->_lower_dimension;}
|
||||||
|
|
||||||
// Allocation profiling support
|
|
||||||
juint alloc_size() const { return _alloc_size; }
|
|
||||||
void set_alloc_size(juint n) { _alloc_size = n; }
|
|
||||||
|
|
||||||
// offset of first element, including any padding for the sake of alignment
|
// offset of first element, including any padding for the sake of alignment
|
||||||
int array_header_in_bytes() const { return layout_helper_header_size(layout_helper()); }
|
int array_header_in_bytes() const { return layout_helper_header_size(layout_helper()); }
|
||||||
int log2_element_size() const { return layout_helper_log2_element_size(layout_helper()); }
|
int log2_element_size() const { return layout_helper_log2_element_size(layout_helper()); }
|
||||||
@ -126,7 +121,6 @@ class ArrayKlass: public Klass {
|
|||||||
// Iterators
|
// Iterators
|
||||||
void array_klasses_do(void f(Klass* k));
|
void array_klasses_do(void f(Klass* k));
|
||||||
void array_klasses_do(void f(Klass* k, TRAPS), TRAPS);
|
void array_klasses_do(void f(Klass* k, TRAPS), TRAPS);
|
||||||
void with_array_klasses_do(void f(Klass* k));
|
|
||||||
|
|
||||||
// GC support
|
// GC support
|
||||||
virtual void oops_do(OopClosure* cl);
|
virtual void oops_do(OopClosure* cl);
|
||||||
|
@ -1321,12 +1321,6 @@ void InstanceKlass::array_klasses_do(void f(Klass* k)) {
|
|||||||
ArrayKlass::cast(array_klasses())->array_klasses_do(f);
|
ArrayKlass::cast(array_klasses())->array_klasses_do(f);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void InstanceKlass::with_array_klasses_do(void f(Klass* k)) {
|
|
||||||
f(this);
|
|
||||||
array_klasses_do(f);
|
|
||||||
}
|
|
||||||
|
|
||||||
#ifdef ASSERT
|
#ifdef ASSERT
|
||||||
static int linear_search(Array<Method*>* methods, Symbol* name, Symbol* signature) {
|
static int linear_search(Array<Method*>* methods, Symbol* name, Symbol* signature) {
|
||||||
int len = methods->length();
|
int len = methods->length();
|
||||||
|
@ -794,7 +794,6 @@ class InstanceKlass: public Klass {
|
|||||||
void methods_do(void f(Method* method));
|
void methods_do(void f(Method* method));
|
||||||
void array_klasses_do(void f(Klass* k));
|
void array_klasses_do(void f(Klass* k));
|
||||||
void array_klasses_do(void f(Klass* k, TRAPS), TRAPS);
|
void array_klasses_do(void f(Klass* k, TRAPS), TRAPS);
|
||||||
void with_array_klasses_do(void f(Klass* k));
|
|
||||||
bool super_types_do(SuperTypeClosure* blk);
|
bool super_types_do(SuperTypeClosure* blk);
|
||||||
|
|
||||||
// Casting from Klass*
|
// Casting from Klass*
|
||||||
@ -874,10 +873,6 @@ class InstanceKlass: public Klass {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Allocation profiling support
|
|
||||||
juint alloc_size() const { return _alloc_count * size_helper(); }
|
|
||||||
void set_alloc_size(juint n) {}
|
|
||||||
|
|
||||||
// Use this to return the size of an instance in heap words:
|
// Use this to return the size of an instance in heap words:
|
||||||
int size_helper() const {
|
int size_helper() const {
|
||||||
return layout_helper_to_size_helper(layout_helper());
|
return layout_helper_to_size_helper(layout_helper());
|
||||||
|
@ -168,7 +168,6 @@ Klass::Klass() {
|
|||||||
set_subklass(NULL);
|
set_subklass(NULL);
|
||||||
set_next_sibling(NULL);
|
set_next_sibling(NULL);
|
||||||
set_next_link(NULL);
|
set_next_link(NULL);
|
||||||
set_alloc_count(0);
|
|
||||||
TRACE_INIT_ID(this);
|
TRACE_INIT_ID(this);
|
||||||
|
|
||||||
set_prototype_header(markOopDesc::prototype());
|
set_prototype_header(markOopDesc::prototype());
|
||||||
@ -543,12 +542,6 @@ Klass* Klass::array_klass_impl(bool or_null, TRAPS) {
|
|||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void Klass::with_array_klasses_do(void f(Klass* k)) {
|
|
||||||
f(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
oop Klass::class_loader() const { return class_loader_data()->class_loader(); }
|
oop Klass::class_loader() const { return class_loader_data()->class_loader(); }
|
||||||
|
|
||||||
const char* Klass::external_name() const {
|
const char* Klass::external_name() const {
|
||||||
|
@ -79,7 +79,6 @@
|
|||||||
// [last_biased_lock_bulk_revocation_time] (64 bits)
|
// [last_biased_lock_bulk_revocation_time] (64 bits)
|
||||||
// [prototype_header]
|
// [prototype_header]
|
||||||
// [biased_lock_revocation_count]
|
// [biased_lock_revocation_count]
|
||||||
// [alloc_count ]
|
|
||||||
// [_modified_oops]
|
// [_modified_oops]
|
||||||
// [_accumulated_modified_oops]
|
// [_accumulated_modified_oops]
|
||||||
// [trace_id]
|
// [trace_id]
|
||||||
@ -171,8 +170,6 @@ class Klass : public Metadata {
|
|||||||
markOop _prototype_header; // Used when biased locking is both enabled and disabled for this type
|
markOop _prototype_header; // Used when biased locking is both enabled and disabled for this type
|
||||||
jint _biased_lock_revocation_count;
|
jint _biased_lock_revocation_count;
|
||||||
|
|
||||||
juint _alloc_count; // allocation profiling support
|
|
||||||
|
|
||||||
TRACE_DEFINE_KLASS_TRACE_ID;
|
TRACE_DEFINE_KLASS_TRACE_ID;
|
||||||
|
|
||||||
// Remembered sets support for the oops in the klasses.
|
// Remembered sets support for the oops in the klasses.
|
||||||
@ -290,11 +287,6 @@ class Klass : public Metadata {
|
|||||||
void set_next_sibling(Klass* s);
|
void set_next_sibling(Klass* s);
|
||||||
|
|
||||||
public:
|
public:
|
||||||
// Allocation profiling support
|
|
||||||
juint alloc_count() const { return _alloc_count; }
|
|
||||||
void set_alloc_count(juint n) { _alloc_count = n; }
|
|
||||||
virtual juint alloc_size() const = 0;
|
|
||||||
virtual void set_alloc_size(juint n) = 0;
|
|
||||||
|
|
||||||
// Compiler support
|
// Compiler support
|
||||||
static ByteSize super_offset() { return in_ByteSize(offset_of(Klass, _super)); }
|
static ByteSize super_offset() { return in_ByteSize(offset_of(Klass, _super)); }
|
||||||
@ -677,7 +669,6 @@ class Klass : public Metadata {
|
|||||||
#endif // INCLUDE_ALL_GCS
|
#endif // INCLUDE_ALL_GCS
|
||||||
|
|
||||||
virtual void array_klasses_do(void f(Klass* k)) {}
|
virtual void array_klasses_do(void f(Klass* k)) {}
|
||||||
virtual void with_array_klasses_do(void f(Klass* k));
|
|
||||||
|
|
||||||
// Return self, except for abstract classes with exactly 1
|
// Return self, except for abstract classes with exactly 1
|
||||||
// implementor. Then return the 1 concrete implementation.
|
// implementor. Then return the 1 concrete implementation.
|
||||||
|
@ -5097,7 +5097,7 @@ _JNI_IMPORT_OR_EXPORT_ jint JNICALL JNI_CreateJavaVM(JavaVM **vm, void **penv, v
|
|||||||
// function used to determine this will always return false. Atomic::xchg
|
// function used to determine this will always return false. Atomic::xchg
|
||||||
// does not have this problem.
|
// does not have this problem.
|
||||||
if (Atomic::xchg(1, &vm_created) == 1) {
|
if (Atomic::xchg(1, &vm_created) == 1) {
|
||||||
return JNI_ERR; // already created, or create attempt in progress
|
return JNI_EEXIST; // already created, or create attempt in progress
|
||||||
}
|
}
|
||||||
if (Atomic::xchg(0, &safe_to_recreate_vm) == 0) {
|
if (Atomic::xchg(0, &safe_to_recreate_vm) == 0) {
|
||||||
return JNI_ERR; // someone tried and failed and retry not allowed.
|
return JNI_ERR; // someone tried and failed and retry not allowed.
|
||||||
@ -5138,9 +5138,21 @@ _JNI_IMPORT_OR_EXPORT_ jint JNICALL JNI_CreateJavaVM(JavaVM **vm, void **penv, v
|
|||||||
event.commit();
|
event.commit();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#ifndef PRODUCT
|
||||||
|
#ifndef TARGET_OS_FAMILY_windows
|
||||||
|
#define CALL_TEST_FUNC_WITH_WRAPPER_IF_NEEDED(f) f()
|
||||||
|
#endif
|
||||||
|
|
||||||
// Check if we should compile all classes on bootclasspath
|
// Check if we should compile all classes on bootclasspath
|
||||||
NOT_PRODUCT(if (CompileTheWorld) ClassLoader::compile_the_world();)
|
if (CompileTheWorld) ClassLoader::compile_the_world();
|
||||||
NOT_PRODUCT(if (ReplayCompiles) ciReplay::replay(thread);)
|
if (ReplayCompiles) ciReplay::replay(thread);
|
||||||
|
|
||||||
|
// Some platforms (like Win*) need a wrapper around these test
|
||||||
|
// functions in order to properly handle error conditions.
|
||||||
|
CALL_TEST_FUNC_WITH_WRAPPER_IF_NEEDED(test_error_handler);
|
||||||
|
CALL_TEST_FUNC_WITH_WRAPPER_IF_NEEDED(execute_internal_vm_tests);
|
||||||
|
#endif
|
||||||
|
|
||||||
// Since this is not a JVM_ENTRY we have to set the thread state manually before leaving.
|
// Since this is not a JVM_ENTRY we have to set the thread state manually before leaving.
|
||||||
ThreadStateTransition::transition_and_fence(thread, _thread_in_vm, _thread_in_native);
|
ThreadStateTransition::transition_and_fence(thread, _thread_in_vm, _thread_in_native);
|
||||||
} else {
|
} else {
|
||||||
@ -5157,8 +5169,6 @@ _JNI_IMPORT_OR_EXPORT_ jint JNICALL JNI_CreateJavaVM(JavaVM **vm, void **penv, v
|
|||||||
OrderAccess::release_store(&vm_created, 0);
|
OrderAccess::release_store(&vm_created, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
NOT_PRODUCT(test_error_handler(ErrorHandlerTest));
|
|
||||||
NOT_PRODUCT(execute_internal_vm_tests());
|
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1121,26 +1121,6 @@ JVM_ENTRY(jobject, JVM_GetProtectionDomain(JNIEnv *env, jclass cls))
|
|||||||
JVM_END
|
JVM_END
|
||||||
|
|
||||||
|
|
||||||
// Obsolete since 1.2 (Class.setProtectionDomain removed), although
|
|
||||||
// still defined in core libraries as of 1.5.
|
|
||||||
JVM_ENTRY(void, JVM_SetProtectionDomain(JNIEnv *env, jclass cls, jobject protection_domain))
|
|
||||||
JVMWrapper("JVM_SetProtectionDomain");
|
|
||||||
if (JNIHandles::resolve(cls) == NULL) {
|
|
||||||
THROW(vmSymbols::java_lang_NullPointerException());
|
|
||||||
}
|
|
||||||
if (!java_lang_Class::is_primitive(JNIHandles::resolve(cls))) {
|
|
||||||
// Call is ignored for primitive types
|
|
||||||
Klass* k = java_lang_Class::as_Klass(JNIHandles::resolve(cls));
|
|
||||||
|
|
||||||
// cls won't be an array, as this called only from ClassLoader.defineClass
|
|
||||||
if (k->oop_is_instance()) {
|
|
||||||
oop pd = JNIHandles::resolve(protection_domain);
|
|
||||||
assert(pd == NULL || pd->is_oop(), "just checking");
|
|
||||||
java_lang_Class::set_protection_domain(k->java_mirror(), pd);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
JVM_END
|
|
||||||
|
|
||||||
static bool is_authorized(Handle context, instanceKlassHandle klass, TRAPS) {
|
static bool is_authorized(Handle context, instanceKlassHandle klass, TRAPS) {
|
||||||
// If there is a security manager and protection domain, check the access
|
// If there is a security manager and protection domain, check the access
|
||||||
// in the protection domain, otherwise it is authorized.
|
// in the protection domain, otherwise it is authorized.
|
||||||
|
@ -471,9 +471,6 @@ JVM_SetClassSigners(JNIEnv *env, jclass cls, jobjectArray signers);
|
|||||||
JNIEXPORT jobject JNICALL
|
JNIEXPORT jobject JNICALL
|
||||||
JVM_GetProtectionDomain(JNIEnv *env, jclass cls);
|
JVM_GetProtectionDomain(JNIEnv *env, jclass cls);
|
||||||
|
|
||||||
JNIEXPORT void JNICALL
|
|
||||||
JVM_SetProtectionDomain(JNIEnv *env, jclass cls, jobject protection_domain);
|
|
||||||
|
|
||||||
JNIEXPORT jboolean JNICALL
|
JNIEXPORT jboolean JNICALL
|
||||||
JVM_IsArrayClass(JNIEnv *env, jclass cls);
|
JVM_IsArrayClass(JNIEnv *env, jclass cls);
|
||||||
|
|
||||||
|
@ -1,143 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved.
|
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
||||||
*
|
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
|
||||||
* under the terms of the GNU General Public License version 2 only, as
|
|
||||||
* published by the Free Software Foundation.
|
|
||||||
*
|
|
||||||
* This code is distributed in the hope that it will be useful, but WITHOUT
|
|
||||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
||||||
* version 2 for more details (a copy is included in the LICENSE file that
|
|
||||||
* accompanied this code).
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License version
|
|
||||||
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
||||||
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
||||||
*
|
|
||||||
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
||||||
* or visit www.oracle.com if you need additional information or have any
|
|
||||||
* questions.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
#include "precompiled.hpp"
|
|
||||||
#include "classfile/systemDictionary.hpp"
|
|
||||||
#include "gc_interface/collectedHeap.inline.hpp"
|
|
||||||
#include "memory/resourceArea.hpp"
|
|
||||||
#include "memory/space.hpp"
|
|
||||||
#include "oops/oop.inline.hpp"
|
|
||||||
#include "oops/oop.inline2.hpp"
|
|
||||||
#include "runtime/aprofiler.hpp"
|
|
||||||
|
|
||||||
|
|
||||||
bool AllocationProfiler::_active = false;
|
|
||||||
GrowableArray<Klass*>* AllocationProfiler::_print_array = NULL;
|
|
||||||
|
|
||||||
|
|
||||||
class AllocProfClosure : public ObjectClosure {
|
|
||||||
public:
|
|
||||||
void do_object(oop obj) {
|
|
||||||
Klass* k = obj->klass();
|
|
||||||
k->set_alloc_count(k->alloc_count() + 1);
|
|
||||||
k->set_alloc_size(k->alloc_size() + obj->size());
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
void AllocationProfiler::iterate_since_last_gc() {
|
|
||||||
if (is_active()) {
|
|
||||||
AllocProfClosure blk;
|
|
||||||
GenCollectedHeap* heap = GenCollectedHeap::heap();
|
|
||||||
heap->object_iterate_since_last_GC(&blk);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void AllocationProfiler::engage() {
|
|
||||||
_active = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void AllocationProfiler::disengage() {
|
|
||||||
_active = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void AllocationProfiler::add_class_to_array(Klass* k) {
|
|
||||||
_print_array->append(k);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void AllocationProfiler::add_classes_to_array(Klass* k) {
|
|
||||||
// Iterate over klass and all array klasses for klass
|
|
||||||
k->with_array_klasses_do(&AllocationProfiler::add_class_to_array);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int AllocationProfiler::compare_classes(Klass** k1, Klass** k2) {
|
|
||||||
// Sort by total allocation size
|
|
||||||
return (*k2)->alloc_size() - (*k1)->alloc_size();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
int AllocationProfiler::average(size_t alloc_size, int alloc_count) {
|
|
||||||
return (int) ((double) (alloc_size * BytesPerWord) / MAX2(alloc_count, 1) + 0.5);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void AllocationProfiler::sort_and_print_array(size_t cutoff) {
|
|
||||||
_print_array->sort(&AllocationProfiler::compare_classes);
|
|
||||||
tty->print_cr("________________Size"
|
|
||||||
"__Instances"
|
|
||||||
"__Average"
|
|
||||||
"__Class________________");
|
|
||||||
size_t total_alloc_size = 0;
|
|
||||||
int total_alloc_count = 0;
|
|
||||||
for (int index = 0; index < _print_array->length(); index++) {
|
|
||||||
Klass* k = _print_array->at(index);
|
|
||||||
size_t alloc_size = k->alloc_size();
|
|
||||||
if (alloc_size > cutoff) {
|
|
||||||
int alloc_count = k->alloc_count();
|
|
||||||
#ifdef PRODUCT
|
|
||||||
const char* name = k->external_name();
|
|
||||||
#else
|
|
||||||
const char* name = k->internal_name();
|
|
||||||
#endif
|
|
||||||
tty->print_cr("%20u %10u %8u %s",
|
|
||||||
alloc_size * BytesPerWord,
|
|
||||||
alloc_count,
|
|
||||||
average(alloc_size, alloc_count),
|
|
||||||
name);
|
|
||||||
total_alloc_size += alloc_size;
|
|
||||||
total_alloc_count += alloc_count;
|
|
||||||
}
|
|
||||||
k->set_alloc_count(0);
|
|
||||||
k->set_alloc_size(0);
|
|
||||||
}
|
|
||||||
tty->print_cr("%20u %10u %8u --total--",
|
|
||||||
total_alloc_size * BytesPerWord,
|
|
||||||
total_alloc_count,
|
|
||||||
average(total_alloc_size, total_alloc_count));
|
|
||||||
tty->cr();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void AllocationProfiler::print(size_t cutoff) {
|
|
||||||
ResourceMark rm;
|
|
||||||
assert(!is_active(), "AllocationProfiler cannot be active while printing profile");
|
|
||||||
|
|
||||||
tty->cr();
|
|
||||||
tty->print_cr("Allocation profile (sizes in bytes, cutoff = " SIZE_FORMAT " bytes):", cutoff * BytesPerWord);
|
|
||||||
tty->cr();
|
|
||||||
|
|
||||||
// Print regular instance klasses and basic type array klasses
|
|
||||||
_print_array = new GrowableArray<Klass*>(SystemDictionary::number_of_classes()*2);
|
|
||||||
SystemDictionary::classes_do(&add_classes_to_array);
|
|
||||||
Universe::basic_type_classes_do(&add_classes_to_array);
|
|
||||||
sort_and_print_array(cutoff);
|
|
||||||
|
|
||||||
// This used to print metadata in the permgen but since there isn't a permgen
|
|
||||||
// anymore, it is not yet implemented.
|
|
||||||
}
|
|
@ -1,71 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 1997, 2012, Oracle and/or its affiliates. All rights reserved.
|
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
||||||
*
|
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
|
||||||
* under the terms of the GNU General Public License version 2 only, as
|
|
||||||
* published by the Free Software Foundation.
|
|
||||||
*
|
|
||||||
* This code is distributed in the hope that it will be useful, but WITHOUT
|
|
||||||
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
|
||||||
* version 2 for more details (a copy is included in the LICENSE file that
|
|
||||||
* accompanied this code).
|
|
||||||
*
|
|
||||||
* You should have received a copy of the GNU General Public License version
|
|
||||||
* 2 along with this work; if not, write to the Free Software Foundation,
|
|
||||||
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
||||||
*
|
|
||||||
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
|
||||||
* or visit www.oracle.com if you need additional information or have any
|
|
||||||
* questions.
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
|
|
||||||
#ifndef SHARE_VM_RUNTIME_APROFILER_HPP
|
|
||||||
#define SHARE_VM_RUNTIME_APROFILER_HPP
|
|
||||||
|
|
||||||
#include "memory/allocation.hpp"
|
|
||||||
#include "memory/universe.hpp"
|
|
||||||
#include "oops/klass.hpp"
|
|
||||||
#include "utilities/top.hpp"
|
|
||||||
|
|
||||||
// A simple allocation profiler for Java. The profiler collects and prints
|
|
||||||
// the number and total size of instances allocated per class, including
|
|
||||||
// array classes.
|
|
||||||
//
|
|
||||||
// The profiler is currently global for all threads. It can be changed to a
|
|
||||||
// per threads profiler by keeping a more elaborate data structure and calling
|
|
||||||
// iterate_since_last_scavenge at thread switches.
|
|
||||||
|
|
||||||
|
|
||||||
class AllocationProfiler: AllStatic {
|
|
||||||
friend class GenCollectedHeap;
|
|
||||||
friend class G1CollectedHeap;
|
|
||||||
friend class MarkSweep;
|
|
||||||
private:
|
|
||||||
static bool _active; // tells whether profiler is active
|
|
||||||
static GrowableArray<Klass*>* _print_array; // temporary array for printing
|
|
||||||
|
|
||||||
// Utility printing functions
|
|
||||||
static void add_class_to_array(Klass* k);
|
|
||||||
static void add_classes_to_array(Klass* k);
|
|
||||||
static int compare_classes(Klass** k1, Klass** k2);
|
|
||||||
static int average(size_t alloc_size, int alloc_count);
|
|
||||||
static void sort_and_print_array(size_t cutoff);
|
|
||||||
|
|
||||||
// Call for collecting allocation information. Called at scavenge, mark-sweep and disengage.
|
|
||||||
static void iterate_since_last_gc();
|
|
||||||
|
|
||||||
public:
|
|
||||||
// Start profiler
|
|
||||||
static void engage();
|
|
||||||
// Stop profiler
|
|
||||||
static void disengage();
|
|
||||||
// Tells whether profiler is active
|
|
||||||
static bool is_active() { return _active; }
|
|
||||||
// Print profile
|
|
||||||
static void print(size_t cutoff); // Cutoff in total allocation size (in words)
|
|
||||||
};
|
|
||||||
|
|
||||||
#endif // SHARE_VM_RUNTIME_APROFILER_HPP
|
|
@ -68,7 +68,6 @@ char* Arguments::_java_command = NULL;
|
|||||||
SystemProperty* Arguments::_system_properties = NULL;
|
SystemProperty* Arguments::_system_properties = NULL;
|
||||||
const char* Arguments::_gc_log_filename = NULL;
|
const char* Arguments::_gc_log_filename = NULL;
|
||||||
bool Arguments::_has_profile = false;
|
bool Arguments::_has_profile = false;
|
||||||
bool Arguments::_has_alloc_profile = false;
|
|
||||||
uintx Arguments::_min_heap_size = 0;
|
uintx Arguments::_min_heap_size = 0;
|
||||||
Arguments::Mode Arguments::_mode = _mixed;
|
Arguments::Mode Arguments::_mode = _mixed;
|
||||||
bool Arguments::_java_compiler = false;
|
bool Arguments::_java_compiler = false;
|
||||||
@ -261,6 +260,9 @@ static ObsoleteFlag obsolete_jvm_flags[] = {
|
|||||||
{ "PrintRevisitStats", JDK_Version::jdk(8), JDK_Version::jdk(9) },
|
{ "PrintRevisitStats", JDK_Version::jdk(8), JDK_Version::jdk(9) },
|
||||||
{ "UseVectoredExceptions", JDK_Version::jdk(8), JDK_Version::jdk(9) },
|
{ "UseVectoredExceptions", JDK_Version::jdk(8), JDK_Version::jdk(9) },
|
||||||
{ "UseSplitVerifier", JDK_Version::jdk(8), JDK_Version::jdk(9) },
|
{ "UseSplitVerifier", JDK_Version::jdk(8), JDK_Version::jdk(9) },
|
||||||
|
{ "UseISM", JDK_Version::jdk(8), JDK_Version::jdk(9) },
|
||||||
|
{ "UsePermISM", JDK_Version::jdk(8), JDK_Version::jdk(9) },
|
||||||
|
{ "UseMPSS", JDK_Version::jdk(8), JDK_Version::jdk(9) },
|
||||||
#ifdef PRODUCT
|
#ifdef PRODUCT
|
||||||
{ "DesiredMethodLimit",
|
{ "DesiredMethodLimit",
|
||||||
JDK_Version::jdk_update(7, 2), JDK_Version::jdk(8) },
|
JDK_Version::jdk_update(7, 2), JDK_Version::jdk(8) },
|
||||||
@ -1855,8 +1857,13 @@ bool Arguments::check_gc_consistency() {
|
|||||||
"please refer to the release notes for the combinations "
|
"please refer to the release notes for the combinations "
|
||||||
"allowed\n");
|
"allowed\n");
|
||||||
status = false;
|
status = false;
|
||||||
|
} else if (ReservedCodeCacheSize > 2*G) {
|
||||||
|
// Code cache size larger than MAXINT is not supported.
|
||||||
|
jio_fprintf(defaultStream::error_stream(),
|
||||||
|
"Invalid ReservedCodeCacheSize=%dM. Must be at most %uM.\n", ReservedCodeCacheSize/M,
|
||||||
|
(2*G)/M);
|
||||||
|
status = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1986,23 +1993,6 @@ bool Arguments::check_vm_args_consistency() {
|
|||||||
status = status && check_gc_consistency();
|
status = status && check_gc_consistency();
|
||||||
status = status && check_stack_pages();
|
status = status && check_stack_pages();
|
||||||
|
|
||||||
if (_has_alloc_profile) {
|
|
||||||
if (UseParallelGC || UseParallelOldGC) {
|
|
||||||
jio_fprintf(defaultStream::error_stream(),
|
|
||||||
"error: invalid argument combination.\n"
|
|
||||||
"Allocation profiling (-Xaprof) cannot be used together with "
|
|
||||||
"Parallel GC (-XX:+UseParallelGC or -XX:+UseParallelOldGC).\n");
|
|
||||||
status = false;
|
|
||||||
}
|
|
||||||
if (UseConcMarkSweepGC) {
|
|
||||||
jio_fprintf(defaultStream::error_stream(),
|
|
||||||
"error: invalid argument combination.\n"
|
|
||||||
"Allocation profiling (-Xaprof) cannot be used together with "
|
|
||||||
"the CMS collector (-XX:+UseConcMarkSweepGC).\n");
|
|
||||||
status = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (CMSIncrementalMode) {
|
if (CMSIncrementalMode) {
|
||||||
if (!UseConcMarkSweepGC) {
|
if (!UseConcMarkSweepGC) {
|
||||||
jio_fprintf(defaultStream::error_stream(),
|
jio_fprintf(defaultStream::error_stream(),
|
||||||
@ -2239,8 +2229,13 @@ bool Arguments::check_vm_args_consistency() {
|
|||||||
"Invalid ReservedCodeCacheSize=%dK. Must be at least %uK.\n", ReservedCodeCacheSize/K,
|
"Invalid ReservedCodeCacheSize=%dK. Must be at least %uK.\n", ReservedCodeCacheSize/K,
|
||||||
min_code_cache_size/K);
|
min_code_cache_size/K);
|
||||||
status = false;
|
status = false;
|
||||||
|
} else if (ReservedCodeCacheSize > 2*G) {
|
||||||
|
// Code cache size larger than MAXINT is not supported.
|
||||||
|
jio_fprintf(defaultStream::error_stream(),
|
||||||
|
"Invalid ReservedCodeCacheSize=%dM. Must be at most %uM.\n", ReservedCodeCacheSize/M,
|
||||||
|
(2*G)/M);
|
||||||
|
status = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
return status;
|
return status;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -2700,9 +2695,6 @@ jint Arguments::parse_each_vm_init_arg(const JavaVMInitArgs* args,
|
|||||||
"Flat profiling is not supported in this VM.\n");
|
"Flat profiling is not supported in this VM.\n");
|
||||||
return JNI_ERR;
|
return JNI_ERR;
|
||||||
#endif // INCLUDE_FPROF
|
#endif // INCLUDE_FPROF
|
||||||
// -Xaprof
|
|
||||||
} else if (match_option(option, "-Xaprof", &tail)) {
|
|
||||||
_has_alloc_profile = true;
|
|
||||||
// -Xconcurrentio
|
// -Xconcurrentio
|
||||||
} else if (match_option(option, "-Xconcurrentio", &tail)) {
|
} else if (match_option(option, "-Xconcurrentio", &tail)) {
|
||||||
FLAG_SET_CMDLINE(bool, UseLWPSynchronization, true);
|
FLAG_SET_CMDLINE(bool, UseLWPSynchronization, true);
|
||||||
@ -2957,13 +2949,6 @@ jint Arguments::parse_each_vm_init_arg(const JavaVMInitArgs* args,
|
|||||||
FLAG_SET_CMDLINE(bool, UseTLAB, true);
|
FLAG_SET_CMDLINE(bool, UseTLAB, true);
|
||||||
} else if (match_option(option, "-XX:-UseTLE", &tail)) {
|
} else if (match_option(option, "-XX:-UseTLE", &tail)) {
|
||||||
FLAG_SET_CMDLINE(bool, UseTLAB, false);
|
FLAG_SET_CMDLINE(bool, UseTLAB, false);
|
||||||
SOLARIS_ONLY(
|
|
||||||
} else if (match_option(option, "-XX:+UsePermISM", &tail)) {
|
|
||||||
warning("-XX:+UsePermISM is obsolete.");
|
|
||||||
FLAG_SET_CMDLINE(bool, UseISM, true);
|
|
||||||
} else if (match_option(option, "-XX:-UsePermISM", &tail)) {
|
|
||||||
FLAG_SET_CMDLINE(bool, UseISM, false);
|
|
||||||
)
|
|
||||||
} else if (match_option(option, "-XX:+DisplayVMOutputToStderr", &tail)) {
|
} else if (match_option(option, "-XX:+DisplayVMOutputToStderr", &tail)) {
|
||||||
FLAG_SET_CMDLINE(bool, DisplayVMOutputToStdout, false);
|
FLAG_SET_CMDLINE(bool, DisplayVMOutputToStdout, false);
|
||||||
FLAG_SET_CMDLINE(bool, DisplayVMOutputToStderr, true);
|
FLAG_SET_CMDLINE(bool, DisplayVMOutputToStderr, true);
|
||||||
@ -3136,8 +3121,6 @@ jint Arguments::finalize_vm_init_args(SysClassPath* scp_p, bool scp_assembly_req
|
|||||||
// Note that large pages are enabled/disabled for both the
|
// Note that large pages are enabled/disabled for both the
|
||||||
// Java heap and the code cache.
|
// Java heap and the code cache.
|
||||||
FLAG_SET_DEFAULT(UseLargePages, false);
|
FLAG_SET_DEFAULT(UseLargePages, false);
|
||||||
SOLARIS_ONLY(FLAG_SET_DEFAULT(UseMPSS, false));
|
|
||||||
SOLARIS_ONLY(FLAG_SET_DEFAULT(UseISM, false));
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Tiered compilation is undefined with C1.
|
// Tiered compilation is undefined with C1.
|
||||||
|
@ -262,7 +262,6 @@ class Arguments : AllStatic {
|
|||||||
|
|
||||||
// Option flags
|
// Option flags
|
||||||
static bool _has_profile;
|
static bool _has_profile;
|
||||||
static bool _has_alloc_profile;
|
|
||||||
static const char* _gc_log_filename;
|
static const char* _gc_log_filename;
|
||||||
static uintx _min_heap_size;
|
static uintx _min_heap_size;
|
||||||
|
|
||||||
@ -464,9 +463,8 @@ class Arguments : AllStatic {
|
|||||||
// -Xloggc:<file>, if not specified will be NULL
|
// -Xloggc:<file>, if not specified will be NULL
|
||||||
static const char* gc_log_filename() { return _gc_log_filename; }
|
static const char* gc_log_filename() { return _gc_log_filename; }
|
||||||
|
|
||||||
// -Xprof/-Xaprof
|
// -Xprof
|
||||||
static bool has_profile() { return _has_profile; }
|
static bool has_profile() { return _has_profile; }
|
||||||
static bool has_alloc_profile() { return _has_alloc_profile; }
|
|
||||||
|
|
||||||
// -Xms, -Xmx
|
// -Xms, -Xmx
|
||||||
static uintx min_heap_size() { return _min_heap_size; }
|
static uintx min_heap_size() { return _min_heap_size; }
|
||||||
|
@ -175,6 +175,7 @@ define_pd_global(intx, InitialCodeCacheSize, 160*K);
|
|||||||
define_pd_global(intx, ReservedCodeCacheSize, 32*M);
|
define_pd_global(intx, ReservedCodeCacheSize, 32*M);
|
||||||
define_pd_global(intx, CodeCacheExpansionSize, 32*K);
|
define_pd_global(intx, CodeCacheExpansionSize, 32*K);
|
||||||
define_pd_global(intx, CodeCacheMinBlockLength, 1);
|
define_pd_global(intx, CodeCacheMinBlockLength, 1);
|
||||||
|
define_pd_global(intx, CodeCacheMinimumUseSpace, 200*K);
|
||||||
define_pd_global(uintx,MetaspaceSize, ScaleForWordSize(4*M));
|
define_pd_global(uintx,MetaspaceSize, ScaleForWordSize(4*M));
|
||||||
define_pd_global(bool, NeverActAsServerClassMachine, true);
|
define_pd_global(bool, NeverActAsServerClassMachine, true);
|
||||||
define_pd_global(uint64_t,MaxRAM, 1ULL*G);
|
define_pd_global(uint64_t,MaxRAM, 1ULL*G);
|
||||||
@ -3679,6 +3680,9 @@ class CommandLineFlags {
|
|||||||
develop(bool, VerifyGenericSignatures, false, \
|
develop(bool, VerifyGenericSignatures, false, \
|
||||||
"Abort VM on erroneous or inconsistent generic signatures") \
|
"Abort VM on erroneous or inconsistent generic signatures") \
|
||||||
\
|
\
|
||||||
|
product(bool, ParseGenericDefaults, false, \
|
||||||
|
"Parse generic signatures for default method handling") \
|
||||||
|
\
|
||||||
product(bool, UseVMInterruptibleIO, false, \
|
product(bool, UseVMInterruptibleIO, false, \
|
||||||
"(Unstable, Solaris-specific) Thread interrupt before or with " \
|
"(Unstable, Solaris-specific) Thread interrupt before or with " \
|
||||||
"EINTR for I/O operations results in OS_INTRPT. The default value"\
|
"EINTR for I/O operations results in OS_INTRPT. The default value"\
|
||||||
|
@ -227,7 +227,7 @@ class HandleArea: public Arena {
|
|||||||
HandleArea* _prev; // link to outer (older) area
|
HandleArea* _prev; // link to outer (older) area
|
||||||
public:
|
public:
|
||||||
// Constructor
|
// Constructor
|
||||||
HandleArea(HandleArea* prev) {
|
HandleArea(HandleArea* prev) : Arena(Chunk::tiny_size) {
|
||||||
debug_only(_handle_mark_nesting = 0);
|
debug_only(_handle_mark_nesting = 0);
|
||||||
debug_only(_no_handle_mark_nesting = 0);
|
debug_only(_no_handle_mark_nesting = 0);
|
||||||
_prev = prev;
|
_prev = prev;
|
||||||
|
@ -42,7 +42,6 @@
|
|||||||
#include "oops/oop.inline.hpp"
|
#include "oops/oop.inline.hpp"
|
||||||
#include "oops/symbol.hpp"
|
#include "oops/symbol.hpp"
|
||||||
#include "prims/jvmtiExport.hpp"
|
#include "prims/jvmtiExport.hpp"
|
||||||
#include "runtime/aprofiler.hpp"
|
|
||||||
#include "runtime/arguments.hpp"
|
#include "runtime/arguments.hpp"
|
||||||
#include "runtime/biasedLocking.hpp"
|
#include "runtime/biasedLocking.hpp"
|
||||||
#include "runtime/compilationPolicy.hpp"
|
#include "runtime/compilationPolicy.hpp"
|
||||||
@ -509,16 +508,6 @@ void before_exit(JavaThread * thread) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
if (Arguments::has_alloc_profile()) {
|
|
||||||
HandleMark hm;
|
|
||||||
// Do one last collection to enumerate all the objects
|
|
||||||
// allocated since the last one.
|
|
||||||
Universe::heap()->collect(GCCause::_allocation_profiler);
|
|
||||||
AllocationProfiler::disengage();
|
|
||||||
AllocationProfiler::print(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (PrintBytecodeHistogram) {
|
if (PrintBytecodeHistogram) {
|
||||||
BytecodeHistogram::print();
|
BytecodeHistogram::print();
|
||||||
}
|
}
|
||||||
|
@ -507,16 +507,16 @@ class os: AllStatic {
|
|||||||
|
|
||||||
// Symbol lookup, find nearest function name; basically it implements
|
// Symbol lookup, find nearest function name; basically it implements
|
||||||
// dladdr() for all platforms. Name of the nearest function is copied
|
// dladdr() for all platforms. Name of the nearest function is copied
|
||||||
// to buf. Distance from its base address is returned as offset.
|
// to buf. Distance from its base address is optionally returned as offset.
|
||||||
// If function name is not found, buf[0] is set to '\0' and offset is
|
// If function name is not found, buf[0] is set to '\0' and offset is
|
||||||
// set to -1.
|
// set to -1 (if offset is non-NULL).
|
||||||
static bool dll_address_to_function_name(address addr, char* buf,
|
static bool dll_address_to_function_name(address addr, char* buf,
|
||||||
int buflen, int* offset);
|
int buflen, int* offset);
|
||||||
|
|
||||||
// Locate DLL/DSO. On success, full path of the library is copied to
|
// Locate DLL/DSO. On success, full path of the library is copied to
|
||||||
// buf, and offset is set to be the distance between addr and the
|
// buf, and offset is optionally set to be the distance between addr
|
||||||
// library's base address. On failure, buf[0] is set to '\0' and
|
// and the library's base address. On failure, buf[0] is set to '\0'
|
||||||
// offset is set to -1.
|
// and offset is set to -1 (if offset is non-NULL).
|
||||||
static bool dll_address_to_library_name(address addr, char* buf,
|
static bool dll_address_to_library_name(address addr, char* buf,
|
||||||
int buflen, int* offset);
|
int buflen, int* offset);
|
||||||
|
|
||||||
|
@ -45,7 +45,6 @@
|
|||||||
#include "prims/jvmtiExport.hpp"
|
#include "prims/jvmtiExport.hpp"
|
||||||
#include "prims/jvmtiThreadState.hpp"
|
#include "prims/jvmtiThreadState.hpp"
|
||||||
#include "prims/privilegedStack.hpp"
|
#include "prims/privilegedStack.hpp"
|
||||||
#include "runtime/aprofiler.hpp"
|
|
||||||
#include "runtime/arguments.hpp"
|
#include "runtime/arguments.hpp"
|
||||||
#include "runtime/biasedLocking.hpp"
|
#include "runtime/biasedLocking.hpp"
|
||||||
#include "runtime/deoptimization.hpp"
|
#include "runtime/deoptimization.hpp"
|
||||||
@ -3677,7 +3676,6 @@ jint Threads::create_vm(JavaVMInitArgs* args, bool* canTryAgain) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (Arguments::has_profile()) FlatProfiler::engage(main_thread, true);
|
if (Arguments::has_profile()) FlatProfiler::engage(main_thread, true);
|
||||||
if (Arguments::has_alloc_profile()) AllocationProfiler::engage();
|
|
||||||
if (MemProfiling) MemProfiler::engage();
|
if (MemProfiling) MemProfiler::engage();
|
||||||
StatSampler::engage();
|
StatSampler::engage();
|
||||||
if (CheckJNICalls) JniPeriodicChecker::engage();
|
if (CheckJNICalls) JniPeriodicChecker::engage();
|
||||||
|
@ -274,7 +274,6 @@ typedef BinaryTreeDictionary<Metablock, FreeList> MetablockTreeDictionary;
|
|||||||
volatile_nonstatic_field(ArrayKlass, _higher_dimension, Klass*) \
|
volatile_nonstatic_field(ArrayKlass, _higher_dimension, Klass*) \
|
||||||
volatile_nonstatic_field(ArrayKlass, _lower_dimension, Klass*) \
|
volatile_nonstatic_field(ArrayKlass, _lower_dimension, Klass*) \
|
||||||
nonstatic_field(ArrayKlass, _vtable_len, int) \
|
nonstatic_field(ArrayKlass, _vtable_len, int) \
|
||||||
nonstatic_field(ArrayKlass, _alloc_size, juint) \
|
|
||||||
nonstatic_field(ArrayKlass, _component_mirror, oop) \
|
nonstatic_field(ArrayKlass, _component_mirror, oop) \
|
||||||
nonstatic_field(CompiledICHolder, _holder_method, Method*) \
|
nonstatic_field(CompiledICHolder, _holder_method, Method*) \
|
||||||
nonstatic_field(CompiledICHolder, _holder_klass, Klass*) \
|
nonstatic_field(CompiledICHolder, _holder_klass, Klass*) \
|
||||||
@ -336,7 +335,6 @@ typedef BinaryTreeDictionary<Metablock, FreeList> MetablockTreeDictionary;
|
|||||||
nonstatic_field(Klass, _access_flags, AccessFlags) \
|
nonstatic_field(Klass, _access_flags, AccessFlags) \
|
||||||
nonstatic_field(Klass, _subklass, Klass*) \
|
nonstatic_field(Klass, _subklass, Klass*) \
|
||||||
nonstatic_field(Klass, _next_sibling, Klass*) \
|
nonstatic_field(Klass, _next_sibling, Klass*) \
|
||||||
nonstatic_field(Klass, _alloc_count, juint) \
|
|
||||||
nonstatic_field(MethodData, _size, int) \
|
nonstatic_field(MethodData, _size, int) \
|
||||||
nonstatic_field(MethodData, _method, Method*) \
|
nonstatic_field(MethodData, _method, Method*) \
|
||||||
nonstatic_field(MethodData, _data_size, int) \
|
nonstatic_field(MethodData, _data_size, int) \
|
||||||
|
@ -486,7 +486,7 @@ int MemBaseline::malloc_sort_by_addr(const void* p1, const void* p2) {
|
|||||||
const MemPointerRecord* mp1 = (const MemPointerRecord*)p1;
|
const MemPointerRecord* mp1 = (const MemPointerRecord*)p1;
|
||||||
const MemPointerRecord* mp2 = (const MemPointerRecord*)p2;
|
const MemPointerRecord* mp2 = (const MemPointerRecord*)p2;
|
||||||
int delta = UNSIGNED_COMPARE(mp1->addr(), mp2->addr());
|
int delta = UNSIGNED_COMPARE(mp1->addr(), mp2->addr());
|
||||||
assert(delta != 0, "dup pointer");
|
assert(p1 == p2 || delta != 0, "dup pointer");
|
||||||
return delta;
|
return delta;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -470,7 +470,21 @@ class MemTracker : AllStatic {
|
|||||||
static void check_NMT_load(Thread* thr) {
|
static void check_NMT_load(Thread* thr) {
|
||||||
assert(thr != NULL, "Sanity check");
|
assert(thr != NULL, "Sanity check");
|
||||||
if (_slowdown_calling_thread && thr != _worker_thread) {
|
if (_slowdown_calling_thread && thr != _worker_thread) {
|
||||||
|
#ifdef _WINDOWS
|
||||||
|
// On Windows, os::NakedYield() does not work as well
|
||||||
|
// as os::yield_all()
|
||||||
os::yield_all();
|
os::yield_all();
|
||||||
|
#else
|
||||||
|
// On Solaris, os::yield_all() depends on os::sleep()
|
||||||
|
// which requires JavaTherad in _thread_in_vm state.
|
||||||
|
// Transits thread to _thread_in_vm state can be dangerous
|
||||||
|
// if caller holds lock, as it may deadlock with Threads_lock.
|
||||||
|
// So use NaKedYield instead.
|
||||||
|
//
|
||||||
|
// Linux and BSD, NakedYield() and yield_all() implementations
|
||||||
|
// are the same.
|
||||||
|
os::NakedYield();
|
||||||
|
#endif
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -314,8 +314,8 @@ bool is_error_reported() {
|
|||||||
#ifndef PRODUCT
|
#ifndef PRODUCT
|
||||||
#include <signal.h>
|
#include <signal.h>
|
||||||
|
|
||||||
void test_error_handler(size_t test_num)
|
void test_error_handler() {
|
||||||
{
|
uintx test_num = ErrorHandlerTest;
|
||||||
if (test_num == 0) return;
|
if (test_num == 0) return;
|
||||||
|
|
||||||
// If asserts are disabled, use the corresponding guarantee instead.
|
// If asserts are disabled, use the corresponding guarantee instead.
|
||||||
@ -327,6 +327,8 @@ void test_error_handler(size_t test_num)
|
|||||||
|
|
||||||
const char* const eol = os::line_separator();
|
const char* const eol = os::line_separator();
|
||||||
const char* const msg = "this message should be truncated during formatting";
|
const char* const msg = "this message should be truncated during formatting";
|
||||||
|
char * const dataPtr = NULL; // bad data pointer
|
||||||
|
const void (*funcPtr)(void) = (const void(*)()) 0xF; // bad function pointer
|
||||||
|
|
||||||
// Keep this in sync with test/runtime/6888954/vmerrors.sh.
|
// Keep this in sync with test/runtime/6888954/vmerrors.sh.
|
||||||
switch (n) {
|
switch (n) {
|
||||||
@ -348,11 +350,16 @@ void test_error_handler(size_t test_num)
|
|||||||
case 9: ShouldNotCallThis();
|
case 9: ShouldNotCallThis();
|
||||||
case 10: ShouldNotReachHere();
|
case 10: ShouldNotReachHere();
|
||||||
case 11: Unimplemented();
|
case 11: Unimplemented();
|
||||||
// This is last because it does not generate an hs_err* file on Windows.
|
// There's no guarantee the bad data pointer will crash us
|
||||||
case 12: os::signal_raise(SIGSEGV);
|
// so "break" out to the ShouldNotReachHere().
|
||||||
|
case 12: *dataPtr = '\0'; break;
|
||||||
|
// There's no guarantee the bad function pointer will crash us
|
||||||
|
// so "break" out to the ShouldNotReachHere().
|
||||||
|
case 13: (*funcPtr)(); break;
|
||||||
|
|
||||||
default: ShouldNotReachHere();
|
default: tty->print_cr("ERROR: %d: unexpected test_num value.", n);
|
||||||
}
|
}
|
||||||
|
ShouldNotReachHere();
|
||||||
}
|
}
|
||||||
#endif // !PRODUCT
|
#endif // !PRODUCT
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 1997, 2012, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 1997, 2013, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -243,7 +243,7 @@ bool is_error_reported();
|
|||||||
void set_error_reported();
|
void set_error_reported();
|
||||||
|
|
||||||
/* Test assert(), fatal(), guarantee(), etc. */
|
/* Test assert(), fatal(), guarantee(), etc. */
|
||||||
NOT_PRODUCT(void test_error_handler(size_t test_num);)
|
NOT_PRODUCT(void test_error_handler();)
|
||||||
|
|
||||||
void pd_ps(frame f);
|
void pd_ps(frame f);
|
||||||
void pd_obfuscate_location(char *buf, size_t buflen);
|
void pd_obfuscate_location(char *buf, size_t buflen);
|
||||||
|
@ -908,10 +908,11 @@ void VMError::report_and_die() {
|
|||||||
// This is not the first error, see if it happened in a different thread
|
// This is not the first error, see if it happened in a different thread
|
||||||
// or in the same thread during error reporting.
|
// or in the same thread during error reporting.
|
||||||
if (first_error_tid != mytid) {
|
if (first_error_tid != mytid) {
|
||||||
jio_snprintf(buffer, sizeof(buffer),
|
char msgbuf[64];
|
||||||
|
jio_snprintf(msgbuf, sizeof(msgbuf),
|
||||||
"[thread " INT64_FORMAT " also had an error]",
|
"[thread " INT64_FORMAT " also had an error]",
|
||||||
mytid);
|
mytid);
|
||||||
out.print_raw_cr(buffer);
|
out.print_raw_cr(msgbuf);
|
||||||
|
|
||||||
// error reporting is not MT-safe, block current thread
|
// error reporting is not MT-safe, block current thread
|
||||||
os::infinite_sleep();
|
os::infinite_sleep();
|
||||||
|
@ -15,7 +15,7 @@
|
|||||||
* @bug 8005956
|
* @bug 8005956
|
||||||
* @summary C2: assert(!def_outside->member(r)) failed: Use of external LRG overlaps the same LRG defined in this block
|
* @summary C2: assert(!def_outside->member(r)) failed: Use of external LRG overlaps the same LRG defined in this block
|
||||||
*
|
*
|
||||||
* @run main PolynomialRoot
|
* @run main/timeout=300 PolynomialRoot
|
||||||
*/
|
*/
|
||||||
|
|
||||||
public class PolynomialRoot {
|
public class PolynomialRoot {
|
||||||
@ -757,18 +757,25 @@ public static int root4(final double [] p,final double [] re_root,final double [
|
|||||||
|
|
||||||
public static void main(final String [] args)
|
public static void main(final String [] args)
|
||||||
{
|
{
|
||||||
|
if (System.getProperty("os.arch").equals("x86") ||
|
||||||
|
System.getProperty("os.arch").equals("amd64") ||
|
||||||
|
System.getProperty("os.arch").equals("x86_64")){
|
||||||
final long t0=System.currentTimeMillis();
|
final long t0=System.currentTimeMillis();
|
||||||
final double eps=1e-6;
|
final double eps=1e-6;
|
||||||
//checkRoots();
|
//checkRoots();
|
||||||
final java.util.Random r=new java.util.Random(-1381923);
|
final java.util.Random r=new java.util.Random(-1381923);
|
||||||
printSpecialValues();
|
printSpecialValues();
|
||||||
|
|
||||||
final int n_tests=10000000;
|
final int n_tests=100000;
|
||||||
//testRoots(2,n_tests,r,eps);
|
//testRoots(2,n_tests,r,eps);
|
||||||
//testRoots(3,n_tests,r,eps);
|
//testRoots(3,n_tests,r,eps);
|
||||||
testRoots(4,n_tests,r,eps);
|
testRoots(4,n_tests,r,eps);
|
||||||
final long t1=System.currentTimeMillis();
|
final long t1=System.currentTimeMillis();
|
||||||
System.err.println("PolynomialRoot.main: "+n_tests+" tests OK done in "+(t1-t0)+" milliseconds. ver=$Id: PolynomialRoot.java,v 1.105 2012/08/18 00:00:05 mal Exp $");
|
System.err.println("PolynomialRoot.main: "+n_tests+" tests OK done in "+(t1-t0)+" milliseconds. ver=$Id: PolynomialRoot.java,v 1.105 2012/08/18 00:00:05 mal Exp $");
|
||||||
|
System.out.println("PASSED");
|
||||||
|
} else {
|
||||||
|
System.out.println("PASS test for non-x86");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
47
hotspot/test/compiler/codecache/CheckUpperLimit.java
Normal file
47
hotspot/test/compiler/codecache/CheckUpperLimit.java
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2013, Oracle and/or its affiliates. All rights reserved.
|
||||||
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
|
*
|
||||||
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU General Public License version 2 only, as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||||||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||||
|
* version 2 for more details (a copy is included in the LICENSE file that
|
||||||
|
* accompanied this code).
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License version
|
||||||
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
*
|
||||||
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||||||
|
* or visit www.oracle.com if you need additional information or have any
|
||||||
|
* questions.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @test
|
||||||
|
* @bug 8015635
|
||||||
|
* @summary Test ensures that the ReservedCodeCacheSize is at most MAXINT
|
||||||
|
* @library /testlibrary
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
import com.oracle.java.testlibrary.*;
|
||||||
|
|
||||||
|
public class CheckUpperLimit {
|
||||||
|
public static void main(String[] args) throws Exception {
|
||||||
|
ProcessBuilder pb;
|
||||||
|
OutputAnalyzer out;
|
||||||
|
|
||||||
|
pb = ProcessTools.createJavaProcessBuilder("-XX:ReservedCodeCacheSize=2048m", "-version");
|
||||||
|
out = new OutputAnalyzer(pb.start());
|
||||||
|
out.shouldHaveExitValue(0);
|
||||||
|
|
||||||
|
pb = ProcessTools.createJavaProcessBuilder("-XX:ReservedCodeCacheSize=2049m", "-version");
|
||||||
|
out = new OutputAnalyzer(pb.start());
|
||||||
|
out.shouldContain("Invalid ReservedCodeCacheSize=");
|
||||||
|
out.shouldHaveExitValue(1);
|
||||||
|
}
|
||||||
|
}
|
@ -1,5 +1,6 @@
|
|||||||
# @test
|
# @test
|
||||||
# @bug 6888954
|
# @bug 6888954
|
||||||
|
# @bug 8015884
|
||||||
# @summary exercise HotSpot error handling code
|
# @summary exercise HotSpot error handling code
|
||||||
# @author John Coomes
|
# @author John Coomes
|
||||||
# @run shell vmerrors.sh
|
# @run shell vmerrors.sh
|
||||||
@ -27,9 +28,24 @@ i=1
|
|||||||
rc=0
|
rc=0
|
||||||
|
|
||||||
assert_re='(assert|guarantee)[(](str|num).*failed: *'
|
assert_re='(assert|guarantee)[(](str|num).*failed: *'
|
||||||
|
# for bad_data_ptr_re:
|
||||||
|
# EXCEPTION_ACCESS_VIOLATION - Win-*
|
||||||
|
# SIGILL - MacOS X
|
||||||
|
# SIGSEGV - Linux-*, Solaris SPARC-*, Solaris X86-*
|
||||||
|
#
|
||||||
|
bad_data_ptr_re='(SIGILL|SIGSEGV|EXCEPTION_ACCESS_VIOLATION).* at pc='
|
||||||
|
#
|
||||||
|
# for bad_func_ptr_re:
|
||||||
|
# EXCEPTION_ACCESS_VIOLATION - Win-*
|
||||||
|
# SIGBUS - Solaris SPARC-64
|
||||||
|
# SIGSEGV - Linux-*, Solaris SPARC-32, Solaris X86-*
|
||||||
|
#
|
||||||
|
# Note: would like to use "pc=0x00*0f," in the pattern, but Solaris SPARC-*
|
||||||
|
# gets its signal at a PC in test_error_handler().
|
||||||
|
#
|
||||||
|
bad_func_ptr_re='(SIGBUS|SIGSEGV|EXCEPTION_ACCESS_VIOLATION).* at pc='
|
||||||
guarantee_re='guarantee[(](str|num).*failed: *'
|
guarantee_re='guarantee[(](str|num).*failed: *'
|
||||||
fatal_re='fatal error: *'
|
fatal_re='fatal error: *'
|
||||||
signal_re='(SIGSEGV|EXCEPTION_ACCESS_VIOLATION).* at pc='
|
|
||||||
tail_1='.*expected null'
|
tail_1='.*expected null'
|
||||||
tail_2='.*num='
|
tail_2='.*num='
|
||||||
|
|
||||||
@ -39,7 +55,8 @@ for re in \
|
|||||||
"${fatal_re}${tail_1}" "${fatal_re}${tail_2}" \
|
"${fatal_re}${tail_1}" "${fatal_re}${tail_2}" \
|
||||||
"${fatal_re}.*truncated" "ChunkPool::allocate" \
|
"${fatal_re}.*truncated" "ChunkPool::allocate" \
|
||||||
"ShouldNotCall" "ShouldNotReachHere" \
|
"ShouldNotCall" "ShouldNotReachHere" \
|
||||||
"Unimplemented" "$signal_re"
|
"Unimplemented" "$bad_data_ptr_re" \
|
||||||
|
"$bad_func_ptr_re"
|
||||||
|
|
||||||
do
|
do
|
||||||
i2=$i
|
i2=$i
|
||||||
|
@ -2,21 +2,21 @@
|
|||||||
|
|
||||||
# Copyright (c) 2013, Oracle and/or its affiliates. All rights reserved.
|
# Copyright (c) 2013, Oracle and/or its affiliates. All rights reserved.
|
||||||
# DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
# DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
|
#
|
||||||
# This code is free software; you can redistribute it and/or modify it
|
# This code is free software; you can redistribute it and/or modify it
|
||||||
# under the terms of the GNU General Public License version 2 only, as
|
# under the terms of the GNU General Public License version 2 only, as
|
||||||
# published by the Free Software Foundation.
|
# published by the Free Software Foundation.
|
||||||
|
#
|
||||||
# This code is distributed in the hope that it will be useful, but WITHOUT
|
# This code is distributed in the hope that it will be useful, but WITHOUT
|
||||||
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
# FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||||
# version 2 for more details (a copy is included in the LICENSE file that
|
# version 2 for more details (a copy is included in the LICENSE file that
|
||||||
# accompanied this code).
|
# accompanied this code).
|
||||||
|
#
|
||||||
# You should have received a copy of the GNU General Public License version
|
# You should have received a copy of the GNU General Public License version
|
||||||
# 2 along with this work; if not, write to the Free Software Foundation,
|
# 2 along with this work; if not, write to the Free Software Foundation,
|
||||||
# Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
# Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
#
|
||||||
# Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
# Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||||||
# or visit www.oracle.com if you need additional information or have any
|
# or visit www.oracle.com if you need additional information or have any
|
||||||
# questions.
|
# questions.
|
||||||
|
@ -219,3 +219,4 @@ a2a2a91075ad85becbe10a39d7fd04ef9bea8df5 jdk8-b92
|
|||||||
42aa9f1828852bb8b77e98ec695211493ae0759d jdk8-b95
|
42aa9f1828852bb8b77e98ec695211493ae0759d jdk8-b95
|
||||||
4a5d3cf2b3af1660db0237e8da324c140e534fa4 jdk8-b96
|
4a5d3cf2b3af1660db0237e8da324c140e534fa4 jdk8-b96
|
||||||
978a95239044f26dcc8a6d59246be07ad6ca6be2 jdk8-b97
|
978a95239044f26dcc8a6d59246be07ad6ca6be2 jdk8-b97
|
||||||
|
c4908732fef5235f1b98cafe0ce507771ef7892c jdk8-b98
|
||||||
|
@ -24,24 +24,23 @@
|
|||||||
*/
|
*/
|
||||||
package java.lang.invoke;
|
package java.lang.invoke;
|
||||||
|
|
||||||
import java.io.Serializable;
|
|
||||||
import java.lang.reflect.Method;
|
|
||||||
import java.lang.reflect.Modifier;
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.Arrays;
|
|
||||||
import java.util.List;
|
|
||||||
import sun.invoke.util.Wrapper;
|
import sun.invoke.util.Wrapper;
|
||||||
import static sun.invoke.util.Wrapper.*;
|
|
||||||
|
import static sun.invoke.util.Wrapper.forPrimitiveType;
|
||||||
|
import static sun.invoke.util.Wrapper.forWrapperType;
|
||||||
|
import static sun.invoke.util.Wrapper.isWrapperType;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Abstract implementation of a lambda metafactory which provides parameter unrolling and input validation.
|
* Abstract implementation of a lambda metafactory which provides parameter
|
||||||
|
* unrolling and input validation.
|
||||||
*
|
*
|
||||||
* @see LambdaMetafactory
|
* @see LambdaMetafactory
|
||||||
*/
|
*/
|
||||||
/* package */ abstract class AbstractValidatingLambdaMetafactory {
|
/* package */ abstract class AbstractValidatingLambdaMetafactory {
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* For context, the comments for the following fields are marked in quotes with their values, given this program:
|
* For context, the comments for the following fields are marked in quotes
|
||||||
|
* with their values, given this program:
|
||||||
* interface II<T> { Object foo(T x); }
|
* interface II<T> { Object foo(T x); }
|
||||||
* interface JJ<R extends Number> extends II<R> { }
|
* interface JJ<R extends Number> extends II<R> { }
|
||||||
* class CC { String impl(int i) { return "impl:"+i; }}
|
* class CC { String impl(int i) { return "impl:"+i; }}
|
||||||
@ -54,9 +53,7 @@ import static sun.invoke.util.Wrapper.*;
|
|||||||
final Class<?> targetClass; // The class calling the meta-factory via invokedynamic "class X"
|
final Class<?> targetClass; // The class calling the meta-factory via invokedynamic "class X"
|
||||||
final MethodType invokedType; // The type of the invoked method "(CC)II"
|
final MethodType invokedType; // The type of the invoked method "(CC)II"
|
||||||
final Class<?> samBase; // The type of the returned instance "interface JJ"
|
final Class<?> samBase; // The type of the returned instance "interface JJ"
|
||||||
final MethodHandle samMethod; // Raw method handle for the functional interface method
|
final String samMethodName; // Name of the SAM method "foo"
|
||||||
final MethodHandleInfo samInfo; // Info about the SAM method handle "MethodHandleInfo[9 II.foo(Object)Object]"
|
|
||||||
final Class<?> samClass; // Interface containing the SAM method "interface II"
|
|
||||||
final MethodType samMethodType; // Type of the SAM method "(Object)Object"
|
final MethodType samMethodType; // Type of the SAM method "(Object)Object"
|
||||||
final MethodHandle implMethod; // Raw method handle for the implementation method
|
final MethodHandle implMethod; // Raw method handle for the implementation method
|
||||||
final MethodHandleInfo implInfo; // Info about the implementation method handle "MethodHandleInfo[5 CC.impl(int)String]"
|
final MethodHandleInfo implInfo; // Info about the implementation method handle "MethodHandleInfo[5 CC.impl(int)String]"
|
||||||
@ -67,44 +64,64 @@ import static sun.invoke.util.Wrapper.*;
|
|||||||
final MethodType instantiatedMethodType; // Instantiated erased functional interface method type "(Integer)Object"
|
final MethodType instantiatedMethodType; // Instantiated erased functional interface method type "(Integer)Object"
|
||||||
final boolean isSerializable; // Should the returned instance be serializable
|
final boolean isSerializable; // Should the returned instance be serializable
|
||||||
final Class<?>[] markerInterfaces; // Additional marker interfaces to be implemented
|
final Class<?>[] markerInterfaces; // Additional marker interfaces to be implemented
|
||||||
|
final MethodType[] additionalBridges; // Signatures of additional methods to bridge
|
||||||
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Meta-factory constructor.
|
* Meta-factory constructor.
|
||||||
*
|
*
|
||||||
* @param caller Stacked automatically by VM; represents a lookup context with the accessibility privileges
|
* @param caller Stacked automatically by VM; represents a lookup context
|
||||||
* of the caller.
|
* with the accessibility privileges of the caller.
|
||||||
* @param invokedType Stacked automatically by VM; the signature of the invoked method, which includes the
|
* @param invokedType Stacked automatically by VM; the signature of the
|
||||||
* expected static type of the returned lambda object, and the static types of the captured
|
* invoked method, which includes the expected static
|
||||||
* arguments for the lambda. In the event that the implementation method is an instance method,
|
* type of the returned lambda object, and the static
|
||||||
* the first argument in the invocation signature will correspond to the receiver.
|
* types of the captured arguments for the lambda. In
|
||||||
* @param samMethod The primary method in the functional interface to which the lambda or method reference is
|
* the event that the implementation method is an
|
||||||
* being converted, represented as a method handle.
|
* instance method, the first argument in the invocation
|
||||||
* @param implMethod The implementation method which should be called (with suitable adaptation of argument
|
* signature will correspond to the receiver.
|
||||||
* types, return types, and adjustment for captured arguments) when methods of the resulting
|
* @param samMethodName Name of the method in the functional interface to
|
||||||
* functional interface instance are invoked.
|
* which the lambda or method reference is being
|
||||||
* @param instantiatedMethodType The signature of the primary functional interface method after type variables
|
* converted, represented as a String.
|
||||||
* are substituted with their instantiation from the capture site
|
* @param samMethodType Type of the method in the functional interface to
|
||||||
|
* which the lambda or method reference is being
|
||||||
|
* converted, represented as a MethodType.
|
||||||
|
* @param implMethod The implementation method which should be called
|
||||||
|
* (with suitable adaptation of argument types, return
|
||||||
|
* types, and adjustment for captured arguments) when
|
||||||
|
* methods of the resulting functional interface instance
|
||||||
|
* are invoked.
|
||||||
|
* @param instantiatedMethodType The signature of the primary functional
|
||||||
|
* interface method after type variables are
|
||||||
|
* substituted with their instantiation from
|
||||||
|
* the capture site
|
||||||
|
* @param isSerializable Should the lambda be made serializable? If set,
|
||||||
|
* either the target type or one of the additional SAM
|
||||||
|
* types must extend {@code Serializable}.
|
||||||
|
* @param markerInterfaces Additional interfaces which the lambda object
|
||||||
|
* should implement.
|
||||||
|
* @param additionalBridges Method types for additional signatures to be
|
||||||
|
* bridged to the implementation method
|
||||||
* @throws ReflectiveOperationException
|
* @throws ReflectiveOperationException
|
||||||
* @throws LambdaConversionException If any of the meta-factory protocol invariants are violated
|
* @throws LambdaConversionException If any of the meta-factory protocol
|
||||||
|
* invariants are violated
|
||||||
*/
|
*/
|
||||||
AbstractValidatingLambdaMetafactory(MethodHandles.Lookup caller,
|
AbstractValidatingLambdaMetafactory(MethodHandles.Lookup caller,
|
||||||
MethodType invokedType,
|
MethodType invokedType,
|
||||||
MethodHandle samMethod,
|
String samMethodName,
|
||||||
|
MethodType samMethodType,
|
||||||
MethodHandle implMethod,
|
MethodHandle implMethod,
|
||||||
MethodType instantiatedMethodType,
|
MethodType instantiatedMethodType,
|
||||||
int flags,
|
boolean isSerializable,
|
||||||
Class<?>[] markerInterfaces)
|
Class<?>[] markerInterfaces,
|
||||||
|
MethodType[] additionalBridges)
|
||||||
throws ReflectiveOperationException, LambdaConversionException {
|
throws ReflectiveOperationException, LambdaConversionException {
|
||||||
this.targetClass = caller.lookupClass();
|
this.targetClass = caller.lookupClass();
|
||||||
this.invokedType = invokedType;
|
this.invokedType = invokedType;
|
||||||
|
|
||||||
this.samBase = invokedType.returnType();
|
this.samBase = invokedType.returnType();
|
||||||
|
|
||||||
this.samMethod = samMethod;
|
this.samMethodName = samMethodName;
|
||||||
this.samInfo = new MethodHandleInfo(samMethod);
|
this.samMethodType = samMethodType;
|
||||||
this.samClass = samInfo.getDeclaringClass();
|
|
||||||
this.samMethodType = samInfo.getMethodType();
|
|
||||||
|
|
||||||
this.implMethod = implMethod;
|
this.implMethod = implMethod;
|
||||||
this.implInfo = new MethodHandleInfo(implMethod);
|
this.implInfo = new MethodHandleInfo(implMethod);
|
||||||
@ -118,32 +135,24 @@ import static sun.invoke.util.Wrapper.*;
|
|||||||
implKind == MethodHandleInfo.REF_invokeInterface;
|
implKind == MethodHandleInfo.REF_invokeInterface;
|
||||||
this.implDefiningClass = implInfo.getDeclaringClass();
|
this.implDefiningClass = implInfo.getDeclaringClass();
|
||||||
this.implMethodType = implInfo.getMethodType();
|
this.implMethodType = implInfo.getMethodType();
|
||||||
|
|
||||||
this.instantiatedMethodType = instantiatedMethodType;
|
this.instantiatedMethodType = instantiatedMethodType;
|
||||||
|
this.isSerializable = isSerializable;
|
||||||
|
this.markerInterfaces = markerInterfaces;
|
||||||
|
this.additionalBridges = additionalBridges;
|
||||||
|
|
||||||
if (!samClass.isInterface()) {
|
if (!samBase.isInterface()) {
|
||||||
throw new LambdaConversionException(String.format(
|
throw new LambdaConversionException(String.format(
|
||||||
"Functional interface %s is not an interface",
|
"Functional interface %s is not an interface",
|
||||||
samClass.getName()));
|
samBase.getName()));
|
||||||
}
|
}
|
||||||
|
|
||||||
boolean foundSerializableSupertype = Serializable.class.isAssignableFrom(samBase);
|
|
||||||
for (Class<?> c : markerInterfaces) {
|
for (Class<?> c : markerInterfaces) {
|
||||||
if (!c.isInterface()) {
|
if (!c.isInterface()) {
|
||||||
throw new LambdaConversionException(String.format(
|
throw new LambdaConversionException(String.format(
|
||||||
"Marker interface %s is not an interface",
|
"Marker interface %s is not an interface",
|
||||||
c.getName()));
|
c.getName()));
|
||||||
}
|
}
|
||||||
foundSerializableSupertype |= Serializable.class.isAssignableFrom(c);
|
|
||||||
}
|
}
|
||||||
this.isSerializable = ((flags & LambdaMetafactory.FLAG_SERIALIZABLE) != 0)
|
|
||||||
|| foundSerializableSupertype;
|
|
||||||
|
|
||||||
if (isSerializable && !foundSerializableSupertype) {
|
|
||||||
markerInterfaces = Arrays.copyOf(markerInterfaces, markerInterfaces.length + 1);
|
|
||||||
markerInterfaces[markerInterfaces.length-1] = Serializable.class;
|
|
||||||
}
|
|
||||||
this.markerInterfaces = markerInterfaces;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -153,20 +162,14 @@ import static sun.invoke.util.Wrapper.*;
|
|||||||
* functional interface
|
* functional interface
|
||||||
* @throws ReflectiveOperationException
|
* @throws ReflectiveOperationException
|
||||||
*/
|
*/
|
||||||
abstract CallSite buildCallSite() throws ReflectiveOperationException, LambdaConversionException;
|
abstract CallSite buildCallSite()
|
||||||
|
throws ReflectiveOperationException, LambdaConversionException;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check the meta-factory arguments for errors
|
* Check the meta-factory arguments for errors
|
||||||
* @throws LambdaConversionException if there are improper conversions
|
* @throws LambdaConversionException if there are improper conversions
|
||||||
*/
|
*/
|
||||||
void validateMetafactoryArgs() throws LambdaConversionException {
|
void validateMetafactoryArgs() throws LambdaConversionException {
|
||||||
// Check target type is a subtype of class where SAM method is defined
|
|
||||||
if (!samClass.isAssignableFrom(samBase)) {
|
|
||||||
throw new LambdaConversionException(
|
|
||||||
String.format("Invalid target type %s for lambda conversion; not a subtype of functional interface %s",
|
|
||||||
samBase.getName(), samClass.getName()));
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (implKind) {
|
switch (implKind) {
|
||||||
case MethodHandleInfo.REF_invokeInterface:
|
case MethodHandleInfo.REF_invokeInterface:
|
||||||
case MethodHandleInfo.REF_invokeVirtual:
|
case MethodHandleInfo.REF_invokeVirtual:
|
||||||
@ -265,9 +268,9 @@ import static sun.invoke.util.Wrapper.*;
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check type adaptability
|
* Check type adaptability for parameter types.
|
||||||
* @param fromType
|
* @param fromType Type to convert from
|
||||||
* @param toType
|
* @param toType Type to convert to
|
||||||
* @param strict If true, do strict checks, else allow that fromType may be parameterized
|
* @param strict If true, do strict checks, else allow that fromType may be parameterized
|
||||||
* @return True if 'fromType' can be passed to an argument of 'toType'
|
* @return True if 'fromType' can be passed to an argument of 'toType'
|
||||||
*/
|
*/
|
||||||
@ -299,15 +302,14 @@ import static sun.invoke.util.Wrapper.*;
|
|||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// both are reference types: fromType should be a superclass of toType.
|
// both are reference types: fromType should be a superclass of toType.
|
||||||
return strict? toType.isAssignableFrom(fromType) : true;
|
return !strict || toType.isAssignableFrom(fromType);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check type adaptability for return types -- special handling of void type) and parameterized fromType
|
* Check type adaptability for return types --
|
||||||
* @param fromType
|
* special handling of void type) and parameterized fromType
|
||||||
* @param toType
|
|
||||||
* @return True if 'fromType' can be converted to 'toType'
|
* @return True if 'fromType' can be converted to 'toType'
|
||||||
*/
|
*/
|
||||||
private boolean isAdaptableToAsReturn(Class<?> fromType, Class<?> toType) {
|
private boolean isAdaptableToAsReturn(Class<?> fromType, Class<?> toType) {
|
||||||
@ -338,89 +340,4 @@ import static sun.invoke.util.Wrapper.*;
|
|||||||
}
|
}
|
||||||
***********************/
|
***********************/
|
||||||
|
|
||||||
/**
|
|
||||||
* Find the functional interface method and corresponding abstract methods
|
|
||||||
* which should be bridged. The functional interface method and those to be
|
|
||||||
* bridged will have the same name and number of parameters. Check for
|
|
||||||
* matching default methods (non-abstract), the VM will create bridges for
|
|
||||||
* default methods; We don't have enough readily available type information
|
|
||||||
* to distinguish between where the functional interface method should be
|
|
||||||
* bridged and where the default method should be bridged; This situation is
|
|
||||||
* flagged.
|
|
||||||
*/
|
|
||||||
class MethodAnalyzer {
|
|
||||||
private final Method[] methods = samBase.getMethods();
|
|
||||||
|
|
||||||
private Method samMethod = null;
|
|
||||||
private final List<Method> methodsToBridge = new ArrayList<>(methods.length);
|
|
||||||
private boolean conflictFoundBetweenDefaultAndBridge = false;
|
|
||||||
|
|
||||||
MethodAnalyzer() {
|
|
||||||
String samMethodName = samInfo.getName();
|
|
||||||
Class<?>[] samParamTypes = samMethodType.parameterArray();
|
|
||||||
int samParamLength = samParamTypes.length;
|
|
||||||
Class<?> samReturnType = samMethodType.returnType();
|
|
||||||
Class<?> objectClass = Object.class;
|
|
||||||
List<Method> defaultMethods = new ArrayList<>(methods.length);
|
|
||||||
|
|
||||||
for (Method m : methods) {
|
|
||||||
if (m.getName().equals(samMethodName) && m.getDeclaringClass() != objectClass) {
|
|
||||||
Class<?>[] mParamTypes = m.getParameterTypes();
|
|
||||||
if (mParamTypes.length == samParamLength) {
|
|
||||||
// Method matches name and parameter length -- and is not Object
|
|
||||||
if (Modifier.isAbstract(m.getModifiers())) {
|
|
||||||
// Method is abstract
|
|
||||||
if (m.getReturnType().equals(samReturnType)
|
|
||||||
&& Arrays.equals(mParamTypes, samParamTypes)) {
|
|
||||||
// Exact match, this is the SAM method signature
|
|
||||||
samMethod = m;
|
|
||||||
} else if (!hasMatchingBridgeSignature(m)) {
|
|
||||||
// Record bridges, exclude methods with duplicate signatures
|
|
||||||
methodsToBridge.add(m);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
// Record default methods for conflict testing
|
|
||||||
defaultMethods.add(m);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for (Method dm : defaultMethods) {
|
|
||||||
if (hasMatchingBridgeSignature(dm)) {
|
|
||||||
conflictFoundBetweenDefaultAndBridge = true;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Method getSamMethod() {
|
|
||||||
return samMethod;
|
|
||||||
}
|
|
||||||
|
|
||||||
List<Method> getMethodsToBridge() {
|
|
||||||
return methodsToBridge;
|
|
||||||
}
|
|
||||||
|
|
||||||
boolean conflictFoundBetweenDefaultAndBridge() {
|
|
||||||
return conflictFoundBetweenDefaultAndBridge;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Search the list of previously found bridge methods to determine if there is a method with the same signature
|
|
||||||
* (return and parameter types) as the specified method.
|
|
||||||
*
|
|
||||||
* @param m The method to match
|
|
||||||
* @return True if the method was found, False otherwise
|
|
||||||
*/
|
|
||||||
private boolean hasMatchingBridgeSignature(Method m) {
|
|
||||||
Class<?>[] ptypes = m.getParameterTypes();
|
|
||||||
Class<?> rtype = m.getReturnType();
|
|
||||||
for (Method md : methodsToBridge) {
|
|
||||||
if (md.getReturnType().equals(rtype) && Arrays.equals(ptypes, md.getParameterTypes())) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
@ -25,22 +25,26 @@
|
|||||||
|
|
||||||
package java.lang.invoke;
|
package java.lang.invoke;
|
||||||
|
|
||||||
import java.lang.reflect.Constructor;
|
|
||||||
import java.lang.reflect.Method;
|
|
||||||
import java.security.ProtectionDomain;
|
|
||||||
import java.util.concurrent.atomic.AtomicInteger;
|
|
||||||
import jdk.internal.org.objectweb.asm.*;
|
import jdk.internal.org.objectweb.asm.*;
|
||||||
import static jdk.internal.org.objectweb.asm.Opcodes.*;
|
|
||||||
import sun.misc.Unsafe;
|
import sun.misc.Unsafe;
|
||||||
|
|
||||||
|
import java.lang.reflect.Constructor;
|
||||||
import java.security.AccessController;
|
import java.security.AccessController;
|
||||||
import java.security.PrivilegedAction;
|
import java.security.PrivilegedAction;
|
||||||
|
import java.security.ProtectionDomain;
|
||||||
|
import java.util.concurrent.atomic.AtomicInteger;
|
||||||
|
|
||||||
|
import static jdk.internal.org.objectweb.asm.Opcodes.*;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Lambda metafactory implementation which dynamically creates an inner-class-like class per lambda callsite.
|
* Lambda metafactory implementation which dynamically creates an
|
||||||
|
* inner-class-like class per lambda callsite.
|
||||||
*
|
*
|
||||||
* @see LambdaMetafactory
|
* @see LambdaMetafactory
|
||||||
*/
|
*/
|
||||||
/* package */ final class InnerClassLambdaMetafactory extends AbstractValidatingLambdaMetafactory {
|
/* package */ final class InnerClassLambdaMetafactory extends AbstractValidatingLambdaMetafactory {
|
||||||
|
private static final Unsafe UNSAFE = Unsafe.getUnsafe();
|
||||||
|
|
||||||
private static final int CLASSFILE_VERSION = 51;
|
private static final int CLASSFILE_VERSION = 51;
|
||||||
private static final String METHOD_DESCRIPTOR_VOID = Type.getMethodDescriptor(Type.VOID_TYPE);
|
private static final String METHOD_DESCRIPTOR_VOID = Type.getMethodDescriptor(Type.VOID_TYPE);
|
||||||
private static final String NAME_MAGIC_ACCESSOR_IMPL = "java/lang/invoke/MagicLambdaImpl";
|
private static final String NAME_MAGIC_ACCESSOR_IMPL = "java/lang/invoke/MagicLambdaImpl";
|
||||||
@ -54,7 +58,7 @@ import java.security.PrivilegedAction;
|
|||||||
private static final String DESCR_CTOR_SERIALIZED_LAMBDA
|
private static final String DESCR_CTOR_SERIALIZED_LAMBDA
|
||||||
= MethodType.methodType(void.class,
|
= MethodType.methodType(void.class,
|
||||||
Class.class,
|
Class.class,
|
||||||
int.class, String.class, String.class, String.class,
|
String.class, String.class, String.class,
|
||||||
int.class, String.class, String.class, String.class,
|
int.class, String.class, String.class, String.class,
|
||||||
String.class,
|
String.class,
|
||||||
Object[].class).toMethodDescriptorString();
|
Object[].class).toMethodDescriptorString();
|
||||||
@ -77,36 +81,56 @@ import java.security.PrivilegedAction;
|
|||||||
private final Type[] instantiatedArgumentTypes; // ASM types for the functional interface arguments
|
private final Type[] instantiatedArgumentTypes; // ASM types for the functional interface arguments
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* General meta-factory constructor, standard cases and allowing for uncommon options such as serialization.
|
* General meta-factory constructor, supporting both standard cases and
|
||||||
|
* allowing for uncommon options such as serialization or bridging.
|
||||||
*
|
*
|
||||||
* @param caller Stacked automatically by VM; represents a lookup context with the accessibility privileges
|
* @param caller Stacked automatically by VM; represents a lookup context
|
||||||
* of the caller.
|
* with the accessibility privileges of the caller.
|
||||||
* @param invokedType Stacked automatically by VM; the signature of the invoked method, which includes the
|
* @param invokedType Stacked automatically by VM; the signature of the
|
||||||
* expected static type of the returned lambda object, and the static types of the captured
|
* invoked method, which includes the expected static
|
||||||
* arguments for the lambda. In the event that the implementation method is an instance method,
|
* type of the returned lambda object, and the static
|
||||||
* the first argument in the invocation signature will correspond to the receiver.
|
* types of the captured arguments for the lambda. In
|
||||||
* @param samMethod The primary method in the functional interface to which the lambda or method reference is
|
* the event that the implementation method is an
|
||||||
* being converted, represented as a method handle.
|
* instance method, the first argument in the invocation
|
||||||
* @param implMethod The implementation method which should be called (with suitable adaptation of argument
|
* signature will correspond to the receiver.
|
||||||
* types, return types, and adjustment for captured arguments) when methods of the resulting
|
* @param samMethodName Name of the method in the functional interface to
|
||||||
* functional interface instance are invoked.
|
* which the lambda or method reference is being
|
||||||
* @param instantiatedMethodType The signature of the primary functional interface method after type variables
|
* converted, represented as a String.
|
||||||
* are substituted with their instantiation from the capture site
|
* @param samMethodType Type of the method in the functional interface to
|
||||||
* @param flags A bitmask containing flags that may influence the translation of this lambda expression. Defined
|
* which the lambda or method reference is being
|
||||||
* fields include FLAG_SERIALIZABLE.
|
* converted, represented as a MethodType.
|
||||||
* @param markerInterfaces Additional interfaces which the lambda object should implement.
|
* @param implMethod The implementation method which should be called (with
|
||||||
|
* suitable adaptation of argument types, return types,
|
||||||
|
* and adjustment for captured arguments) when methods of
|
||||||
|
* the resulting functional interface instance are invoked.
|
||||||
|
* @param instantiatedMethodType The signature of the primary functional
|
||||||
|
* interface method after type variables are
|
||||||
|
* substituted with their instantiation from
|
||||||
|
* the capture site
|
||||||
|
* @param isSerializable Should the lambda be made serializable? If set,
|
||||||
|
* either the target type or one of the additional SAM
|
||||||
|
* types must extend {@code Serializable}.
|
||||||
|
* @param markerInterfaces Additional interfaces which the lambda object
|
||||||
|
* should implement.
|
||||||
|
* @param additionalBridges Method types for additional signatures to be
|
||||||
|
* bridged to the implementation method
|
||||||
* @throws ReflectiveOperationException
|
* @throws ReflectiveOperationException
|
||||||
* @throws LambdaConversionException If any of the meta-factory protocol invariants are violated
|
* @throws LambdaConversionException If any of the meta-factory protocol
|
||||||
|
* invariants are violated
|
||||||
*/
|
*/
|
||||||
public InnerClassLambdaMetafactory(MethodHandles.Lookup caller,
|
public InnerClassLambdaMetafactory(MethodHandles.Lookup caller,
|
||||||
MethodType invokedType,
|
MethodType invokedType,
|
||||||
MethodHandle samMethod,
|
String samMethodName,
|
||||||
|
MethodType samMethodType,
|
||||||
MethodHandle implMethod,
|
MethodHandle implMethod,
|
||||||
MethodType instantiatedMethodType,
|
MethodType instantiatedMethodType,
|
||||||
int flags,
|
boolean isSerializable,
|
||||||
Class<?>[] markerInterfaces)
|
Class<?>[] markerInterfaces,
|
||||||
|
MethodType[] additionalBridges)
|
||||||
throws ReflectiveOperationException, LambdaConversionException {
|
throws ReflectiveOperationException, LambdaConversionException {
|
||||||
super(caller, invokedType, samMethod, implMethod, instantiatedMethodType, flags, markerInterfaces);
|
super(caller, invokedType, samMethodName, samMethodType,
|
||||||
|
implMethod, instantiatedMethodType,
|
||||||
|
isSerializable, markerInterfaces, additionalBridges);
|
||||||
implMethodClassName = implDefiningClass.getName().replace('.', '/');
|
implMethodClassName = implDefiningClass.getName().replace('.', '/');
|
||||||
implMethodName = implInfo.getName();
|
implMethodName = implInfo.getName();
|
||||||
implMethodDesc = implMethodType.toMethodDescriptorString();
|
implMethodDesc = implMethodType.toMethodDescriptorString();
|
||||||
@ -124,7 +148,8 @@ import java.security.PrivilegedAction;
|
|||||||
for (int i = 0; i < argTypes.length; i++) {
|
for (int i = 0; i < argTypes.length; i++) {
|
||||||
argNames[i] = "arg$" + (i + 1);
|
argNames[i] = "arg$" + (i + 1);
|
||||||
}
|
}
|
||||||
instantiatedArgumentTypes = Type.getArgumentTypes(instantiatedMethodType.toMethodDescriptorString());
|
instantiatedArgumentTypes = Type.getArgumentTypes(
|
||||||
|
instantiatedMethodType.toMethodDescriptorString());
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -136,7 +161,8 @@ import java.security.PrivilegedAction;
|
|||||||
* @return a CallSite, which, when invoked, will return an instance of the
|
* @return a CallSite, which, when invoked, will return an instance of the
|
||||||
* functional interface
|
* functional interface
|
||||||
* @throws ReflectiveOperationException
|
* @throws ReflectiveOperationException
|
||||||
* @throws LambdaConversionException If properly formed functional interface is not found
|
* @throws LambdaConversionException If properly formed functional interface
|
||||||
|
* is not found
|
||||||
*/
|
*/
|
||||||
@Override
|
@Override
|
||||||
CallSite buildCallSite() throws ReflectiveOperationException, LambdaConversionException {
|
CallSite buildCallSite() throws ReflectiveOperationException, LambdaConversionException {
|
||||||
@ -176,13 +202,20 @@ import java.security.PrivilegedAction;
|
|||||||
* Generate a class file which implements the functional
|
* Generate a class file which implements the functional
|
||||||
* interface, define and return the class.
|
* interface, define and return the class.
|
||||||
*
|
*
|
||||||
|
* @implNote The class that is generated does not include signature
|
||||||
|
* information for exceptions that may be present on the SAM method.
|
||||||
|
* This is to reduce classfile size, and is harmless as checked exceptions
|
||||||
|
* are erased anyway, no one will ever compile against this classfile,
|
||||||
|
* and we make no guarantees about the reflective properties of lambda
|
||||||
|
* objects.
|
||||||
|
*
|
||||||
* @return a Class which implements the functional interface
|
* @return a Class which implements the functional interface
|
||||||
* @throws LambdaConversionException If properly formed functional interface is not found
|
* @throws LambdaConversionException If properly formed functional interface
|
||||||
|
* is not found
|
||||||
*/
|
*/
|
||||||
private Class<?> spinInnerClass() throws LambdaConversionException {
|
private Class<?> spinInnerClass() throws LambdaConversionException {
|
||||||
String samName = samBase.getName().replace('.', '/');
|
|
||||||
String[] interfaces = new String[markerInterfaces.length + 1];
|
String[] interfaces = new String[markerInterfaces.length + 1];
|
||||||
interfaces[0] = samName;
|
interfaces[0] = samBase.getName().replace('.', '/');
|
||||||
for (int i=0; i<markerInterfaces.length; i++) {
|
for (int i=0; i<markerInterfaces.length; i++) {
|
||||||
interfaces[i+1] = markerInterfaces[i].getName().replace('.', '/');
|
interfaces[i+1] = markerInterfaces[i].getName().replace('.', '/');
|
||||||
}
|
}
|
||||||
@ -192,35 +225,33 @@ import java.security.PrivilegedAction;
|
|||||||
|
|
||||||
// Generate final fields to be filled in by constructor
|
// Generate final fields to be filled in by constructor
|
||||||
for (int i = 0; i < argTypes.length; i++) {
|
for (int i = 0; i < argTypes.length; i++) {
|
||||||
FieldVisitor fv = cw.visitField(ACC_PRIVATE + ACC_FINAL, argNames[i], argTypes[i].getDescriptor(),
|
FieldVisitor fv = cw.visitField(ACC_PRIVATE + ACC_FINAL,
|
||||||
|
argNames[i],
|
||||||
|
argTypes[i].getDescriptor(),
|
||||||
null, null);
|
null, null);
|
||||||
fv.visitEnd();
|
fv.visitEnd();
|
||||||
}
|
}
|
||||||
|
|
||||||
generateConstructor();
|
generateConstructor();
|
||||||
|
|
||||||
MethodAnalyzer ma = new MethodAnalyzer();
|
|
||||||
|
|
||||||
// Forward the SAM method
|
// Forward the SAM method
|
||||||
if (ma.getSamMethod() == null) {
|
String methodDescriptor = samMethodType.toMethodDescriptorString();
|
||||||
throw new LambdaConversionException(String.format("Functional interface method not found: %s", samMethodType));
|
MethodVisitor mv = cw.visitMethod(ACC_PUBLIC, samMethodName,
|
||||||
} else {
|
methodDescriptor, null, null);
|
||||||
generateForwardingMethod(ma.getSamMethod(), false);
|
new ForwardingMethodGenerator(mv).generate(methodDescriptor);
|
||||||
}
|
|
||||||
|
|
||||||
// Forward the bridges
|
// Forward the bridges
|
||||||
// @@@ The commented-out code is temporary, pending the VM's ability to bridge all methods on request
|
if (additionalBridges != null) {
|
||||||
// @@@ Once the VM can do fail-over, uncomment the !ma.wasDefaultMethodFound() test, and emit the appropriate
|
for (MethodType mt : additionalBridges) {
|
||||||
// @@@ classfile attribute to request custom bridging. See 8002092.
|
methodDescriptor = mt.toMethodDescriptorString();
|
||||||
if (!ma.getMethodsToBridge().isEmpty() /* && !ma.conflictFoundBetweenDefaultAndBridge() */ ) {
|
mv = cw.visitMethod(ACC_PUBLIC|ACC_BRIDGE, samMethodName,
|
||||||
for (Method m : ma.getMethodsToBridge()) {
|
methodDescriptor, null, null);
|
||||||
generateForwardingMethod(m, true);
|
new ForwardingMethodGenerator(mv).generate(methodDescriptor);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isSerializable) {
|
if (isSerializable)
|
||||||
generateWriteReplace();
|
generateWriteReplace();
|
||||||
}
|
|
||||||
|
|
||||||
cw.visitEnd();
|
cw.visitEnd();
|
||||||
|
|
||||||
@ -229,11 +260,14 @@ import java.security.PrivilegedAction;
|
|||||||
final byte[] classBytes = cw.toByteArray();
|
final byte[] classBytes = cw.toByteArray();
|
||||||
|
|
||||||
/*** Uncomment to dump the generated file
|
/*** Uncomment to dump the generated file
|
||||||
System.out.printf("Loaded: %s (%d bytes) %n", lambdaClassName, classBytes.length);
|
System.out.printf("Loaded: %s (%d bytes) %n", lambdaClassName,
|
||||||
try (FileOutputStream fos = new FileOutputStream(lambdaClassName.replace('/', '.') + ".class")) {
|
classBytes.length);
|
||||||
|
try (FileOutputStream fos = new FileOutputStream(lambdaClassName
|
||||||
|
.replace('/', '.') + ".class")) {
|
||||||
fos.write(classBytes);
|
fos.write(classBytes);
|
||||||
} catch (IOException ex) {
|
} catch (IOException ex) {
|
||||||
PlatformLogger.getLogger(InnerClassLambdaMetafactory.class.getName()).severe(ex.getMessage(), ex);
|
PlatformLogger.getLogger(InnerClassLambdaMetafactory.class
|
||||||
|
.getName()).severe(ex.getMessage(), ex);
|
||||||
}
|
}
|
||||||
***/
|
***/
|
||||||
|
|
||||||
@ -249,7 +283,8 @@ import java.security.PrivilegedAction;
|
|||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
return (Class<?>) Unsafe.getUnsafe().defineClass(lambdaClassName, classBytes, 0, classBytes.length,
|
return UNSAFE.defineClass(lambdaClassName,
|
||||||
|
classBytes, 0, classBytes.length,
|
||||||
loader, pd);
|
loader, pd);
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -258,19 +293,23 @@ import java.security.PrivilegedAction;
|
|||||||
*/
|
*/
|
||||||
private void generateConstructor() {
|
private void generateConstructor() {
|
||||||
// Generate constructor
|
// Generate constructor
|
||||||
MethodVisitor ctor = cw.visitMethod(ACC_PRIVATE, NAME_CTOR, constructorDesc, null, null);
|
MethodVisitor ctor = cw.visitMethod(ACC_PRIVATE, NAME_CTOR,
|
||||||
|
constructorDesc, null, null);
|
||||||
ctor.visitCode();
|
ctor.visitCode();
|
||||||
ctor.visitVarInsn(ALOAD, 0);
|
ctor.visitVarInsn(ALOAD, 0);
|
||||||
ctor.visitMethodInsn(INVOKESPECIAL, NAME_MAGIC_ACCESSOR_IMPL, NAME_CTOR, METHOD_DESCRIPTOR_VOID);
|
ctor.visitMethodInsn(INVOKESPECIAL, NAME_MAGIC_ACCESSOR_IMPL, NAME_CTOR,
|
||||||
|
METHOD_DESCRIPTOR_VOID);
|
||||||
int lvIndex = 0;
|
int lvIndex = 0;
|
||||||
for (int i = 0; i < argTypes.length; i++) {
|
for (int i = 0; i < argTypes.length; i++) {
|
||||||
ctor.visitVarInsn(ALOAD, 0);
|
ctor.visitVarInsn(ALOAD, 0);
|
||||||
ctor.visitVarInsn(argTypes[i].getOpcode(ILOAD), lvIndex + 1);
|
ctor.visitVarInsn(argTypes[i].getOpcode(ILOAD), lvIndex + 1);
|
||||||
lvIndex += argTypes[i].getSize();
|
lvIndex += argTypes[i].getSize();
|
||||||
ctor.visitFieldInsn(PUTFIELD, lambdaClassName, argNames[i], argTypes[i].getDescriptor());
|
ctor.visitFieldInsn(PUTFIELD, lambdaClassName, argNames[i],
|
||||||
|
argTypes[i].getDescriptor());
|
||||||
}
|
}
|
||||||
ctor.visitInsn(RETURN);
|
ctor.visitInsn(RETURN);
|
||||||
ctor.visitMaxs(-1, -1); // Maxs computed by ClassWriter.COMPUTE_MAXS, these arguments ignored
|
// Maxs computed by ClassWriter.COMPUTE_MAXS, these arguments ignored
|
||||||
|
ctor.visitMaxs(-1, -1);
|
||||||
ctor.visitEnd();
|
ctor.visitEnd();
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -279,18 +318,18 @@ import java.security.PrivilegedAction;
|
|||||||
*/
|
*/
|
||||||
private void generateWriteReplace() {
|
private void generateWriteReplace() {
|
||||||
TypeConvertingMethodAdapter mv
|
TypeConvertingMethodAdapter mv
|
||||||
= new TypeConvertingMethodAdapter(cw.visitMethod(ACC_PRIVATE + ACC_FINAL,
|
= new TypeConvertingMethodAdapter(
|
||||||
|
cw.visitMethod(ACC_PRIVATE + ACC_FINAL,
|
||||||
NAME_METHOD_WRITE_REPLACE, DESCR_METHOD_WRITE_REPLACE,
|
NAME_METHOD_WRITE_REPLACE, DESCR_METHOD_WRITE_REPLACE,
|
||||||
null, null));
|
null, null));
|
||||||
|
|
||||||
mv.visitCode();
|
mv.visitCode();
|
||||||
mv.visitTypeInsn(NEW, NAME_SERIALIZED_LAMBDA);
|
mv.visitTypeInsn(NEW, NAME_SERIALIZED_LAMBDA);
|
||||||
mv.visitInsn(DUP);;
|
mv.visitInsn(DUP);
|
||||||
mv.visitLdcInsn(Type.getType(targetClass));
|
mv.visitLdcInsn(Type.getType(targetClass));
|
||||||
mv.visitLdcInsn(samInfo.getReferenceKind());
|
|
||||||
mv.visitLdcInsn(invokedType.returnType().getName().replace('.', '/'));
|
mv.visitLdcInsn(invokedType.returnType().getName().replace('.', '/'));
|
||||||
mv.visitLdcInsn(samInfo.getName());
|
mv.visitLdcInsn(samMethodName);
|
||||||
mv.visitLdcInsn(samInfo.getMethodType().toMethodDescriptorString());
|
mv.visitLdcInsn(samMethodType.toMethodDescriptorString());
|
||||||
mv.visitLdcInsn(implInfo.getReferenceKind());
|
mv.visitLdcInsn(implInfo.getReferenceKind());
|
||||||
mv.visitLdcInsn(implInfo.getDeclaringClass().getName().replace('.', '/'));
|
mv.visitLdcInsn(implInfo.getDeclaringClass().getName().replace('.', '/'));
|
||||||
mv.visitLdcInsn(implInfo.getName());
|
mv.visitLdcInsn(implInfo.getName());
|
||||||
@ -303,35 +342,19 @@ import java.security.PrivilegedAction;
|
|||||||
mv.visitInsn(DUP);
|
mv.visitInsn(DUP);
|
||||||
mv.iconst(i);
|
mv.iconst(i);
|
||||||
mv.visitVarInsn(ALOAD, 0);
|
mv.visitVarInsn(ALOAD, 0);
|
||||||
mv.visitFieldInsn(GETFIELD, lambdaClassName, argNames[i], argTypes[i].getDescriptor());
|
mv.visitFieldInsn(GETFIELD, lambdaClassName, argNames[i],
|
||||||
|
argTypes[i].getDescriptor());
|
||||||
mv.boxIfTypePrimitive(argTypes[i]);
|
mv.boxIfTypePrimitive(argTypes[i]);
|
||||||
mv.visitInsn(AASTORE);
|
mv.visitInsn(AASTORE);
|
||||||
}
|
}
|
||||||
mv.visitMethodInsn(INVOKESPECIAL, NAME_SERIALIZED_LAMBDA, NAME_CTOR,
|
mv.visitMethodInsn(INVOKESPECIAL, NAME_SERIALIZED_LAMBDA, NAME_CTOR,
|
||||||
DESCR_CTOR_SERIALIZED_LAMBDA);
|
DESCR_CTOR_SERIALIZED_LAMBDA);
|
||||||
mv.visitInsn(ARETURN);
|
mv.visitInsn(ARETURN);
|
||||||
mv.visitMaxs(-1, -1); // Maxs computed by ClassWriter.COMPUTE_MAXS, these arguments ignored
|
// Maxs computed by ClassWriter.COMPUTE_MAXS, these arguments ignored
|
||||||
|
mv.visitMaxs(-1, -1);
|
||||||
mv.visitEnd();
|
mv.visitEnd();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Generate a method which calls the lambda implementation method,
|
|
||||||
* converting arguments, as needed.
|
|
||||||
* @param m The method whose signature should be generated
|
|
||||||
* @param isBridge True if this methods should be flagged as a bridge
|
|
||||||
*/
|
|
||||||
private void generateForwardingMethod(Method m, boolean isBridge) {
|
|
||||||
Class<?>[] exceptionTypes = m.getExceptionTypes();
|
|
||||||
String[] exceptionNames = new String[exceptionTypes.length];
|
|
||||||
for (int i = 0; i < exceptionTypes.length; i++) {
|
|
||||||
exceptionNames[i] = exceptionTypes[i].getName().replace('.', '/');
|
|
||||||
}
|
|
||||||
String methodDescriptor = Type.getMethodDescriptor(m);
|
|
||||||
int access = isBridge? ACC_PUBLIC | ACC_BRIDGE : ACC_PUBLIC;
|
|
||||||
MethodVisitor mv = cw.visitMethod(access, m.getName(), methodDescriptor, null, exceptionNames);
|
|
||||||
new ForwardingMethodGenerator(mv).generate(m);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* This class generates a method body which calls the lambda implementation
|
* This class generates a method body which calls the lambda implementation
|
||||||
* method, converting arguments, as needed.
|
* method, converting arguments, as needed.
|
||||||
@ -342,36 +365,39 @@ import java.security.PrivilegedAction;
|
|||||||
super(mv);
|
super(mv);
|
||||||
}
|
}
|
||||||
|
|
||||||
void generate(Method m) throws InternalError {
|
void generate(String methodDescriptor) {
|
||||||
visitCode();
|
visitCode();
|
||||||
|
|
||||||
if (implKind == MethodHandleInfo.REF_newInvokeSpecial) {
|
if (implKind == MethodHandleInfo.REF_newInvokeSpecial) {
|
||||||
visitTypeInsn(NEW, implMethodClassName);
|
visitTypeInsn(NEW, implMethodClassName);
|
||||||
visitInsn(DUP);;
|
visitInsn(DUP);
|
||||||
}
|
}
|
||||||
for (int i = 0; i < argTypes.length; i++) {
|
for (int i = 0; i < argTypes.length; i++) {
|
||||||
visitVarInsn(ALOAD, 0);
|
visitVarInsn(ALOAD, 0);
|
||||||
visitFieldInsn(GETFIELD, lambdaClassName, argNames[i], argTypes[i].getDescriptor());
|
visitFieldInsn(GETFIELD, lambdaClassName, argNames[i],
|
||||||
|
argTypes[i].getDescriptor());
|
||||||
}
|
}
|
||||||
|
|
||||||
convertArgumentTypes(Type.getArgumentTypes(m));
|
convertArgumentTypes(Type.getArgumentTypes(methodDescriptor));
|
||||||
|
|
||||||
// Invoke the method we want to forward to
|
// Invoke the method we want to forward to
|
||||||
visitMethodInsn(invocationOpcode(), implMethodClassName, implMethodName, implMethodDesc);
|
visitMethodInsn(invocationOpcode(), implMethodClassName, implMethodName, implMethodDesc);
|
||||||
|
|
||||||
// Convert the return value (if any) and return it
|
// Convert the return value (if any) and return it
|
||||||
// Note: if adapting from non-void to void, the 'return' instruction will pop the unneeded result
|
// Note: if adapting from non-void to void, the 'return'
|
||||||
Type samReturnType = Type.getReturnType(m);
|
// instruction will pop the unneeded result
|
||||||
|
Type samReturnType = Type.getReturnType(methodDescriptor);
|
||||||
convertType(implMethodReturnType, samReturnType, samReturnType);
|
convertType(implMethodReturnType, samReturnType, samReturnType);
|
||||||
visitInsn(samReturnType.getOpcode(Opcodes.IRETURN));
|
visitInsn(samReturnType.getOpcode(Opcodes.IRETURN));
|
||||||
|
// Maxs computed by ClassWriter.COMPUTE_MAXS,these arguments ignored
|
||||||
visitMaxs(-1, -1); // Maxs computed by ClassWriter.COMPUTE_MAXS, these arguments ignored
|
visitMaxs(-1, -1);
|
||||||
visitEnd();
|
visitEnd();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void convertArgumentTypes(Type[] samArgumentTypes) {
|
private void convertArgumentTypes(Type[] samArgumentTypes) {
|
||||||
int lvIndex = 0;
|
int lvIndex = 0;
|
||||||
boolean samIncludesReceiver = implIsInstanceMethod && argTypes.length == 0;
|
boolean samIncludesReceiver = implIsInstanceMethod &&
|
||||||
|
argTypes.length == 0;
|
||||||
int samReceiverLength = samIncludesReceiver ? 1 : 0;
|
int samReceiverLength = samIncludesReceiver ? 1 : 0;
|
||||||
if (samIncludesReceiver) {
|
if (samIncludesReceiver) {
|
||||||
// push receiver
|
// push receiver
|
||||||
@ -395,7 +421,9 @@ import java.security.PrivilegedAction;
|
|||||||
}
|
}
|
||||||
|
|
||||||
private void convertType(Type argType, Type targetType, Type functionalType) {
|
private void convertType(Type argType, Type targetType, Type functionalType) {
|
||||||
convertType(argType.getDescriptor(), targetType.getDescriptor(), functionalType.getDescriptor());
|
convertType(argType.getDescriptor(),
|
||||||
|
targetType.getDescriptor(),
|
||||||
|
functionalType.getDescriptor());
|
||||||
}
|
}
|
||||||
|
|
||||||
private int invocationOpcode() throws InternalError {
|
private int invocationOpcode() throws InternalError {
|
||||||
|
@ -25,6 +25,9 @@
|
|||||||
|
|
||||||
package java.lang.invoke;
|
package java.lang.invoke;
|
||||||
|
|
||||||
|
import java.io.Serializable;
|
||||||
|
import java.util.Arrays;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* <p>Bootstrap methods for converting lambda expressions and method references to functional interface objects.</p>
|
* <p>Bootstrap methods for converting lambda expressions and method references to functional interface objects.</p>
|
||||||
*
|
*
|
||||||
@ -44,16 +47,11 @@ package java.lang.invoke;
|
|||||||
*
|
*
|
||||||
* <p>When parameterized types are used, the instantiated type of the functional interface method may be different
|
* <p>When parameterized types are used, the instantiated type of the functional interface method may be different
|
||||||
* from that in the functional interface. For example, consider
|
* from that in the functional interface. For example, consider
|
||||||
* <code>interface I<T> { int m(T x); }</code> if this functional interface type is used in a lambda
|
* {@code interface I<T> { int m(T x); }} if this functional interface type is used in a lambda
|
||||||
* <code>I<Byte> v = ...</code>, we need both the actual functional interface method which has the signature
|
* {@code I<Byte>; v = ...}, we need both the actual functional interface method which has the signature
|
||||||
* <code>(Object)int</code> and the erased instantiated type of the functional interface method (or simply
|
* {@code (Object)int} and the erased instantiated type of the functional interface method (or simply
|
||||||
* <I>instantiated method type</I>), which has signature
|
* <I>instantiated method type</I>), which has signature
|
||||||
* <code>(Byte)int</code>.
|
* {@code (Byte)int}.
|
||||||
*
|
|
||||||
* <p>While functional interfaces only have a single abstract method from the language perspective (concrete
|
|
||||||
* methods in Object are and default methods may be present), at the bytecode level they may actually have multiple
|
|
||||||
* methods because of the need for bridge methods. Invoking any of these methods on the lambda object will result
|
|
||||||
* in invoking the implementation method.
|
|
||||||
*
|
*
|
||||||
* <p>The argument list of the implementation method and the argument list of the functional interface method(s)
|
* <p>The argument list of the implementation method and the argument list of the functional interface method(s)
|
||||||
* may differ in several ways. The implementation methods may have additional arguments to accommodate arguments
|
* may differ in several ways. The implementation methods may have additional arguments to accommodate arguments
|
||||||
@ -137,108 +135,147 @@ package java.lang.invoke;
|
|||||||
* </tr>
|
* </tr>
|
||||||
* </table>
|
* </table>
|
||||||
*
|
*
|
||||||
* The default bootstrap ({@link #metaFactory}) represents the common cases and uses an optimized protocol.
|
* The default bootstrap ({@link #metafactory}) represents the common cases and uses an optimized protocol.
|
||||||
* Alternate bootstraps (e.g., {@link #altMetaFactory}) exist to support uncommon cases such as serialization
|
* Alternate bootstraps (e.g., {@link #altMetafactory}) exist to support uncommon cases such as serialization
|
||||||
* or additional marker superinterfaces.
|
* or additional marker superinterfaces.
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
public class LambdaMetafactory {
|
public class LambdaMetafactory {
|
||||||
|
|
||||||
/** Flag for alternate metafactories indicating the lambda object is must to be serializable */
|
/** Flag for alternate metafactories indicating the lambda object is
|
||||||
|
* must to be serializable */
|
||||||
public static final int FLAG_SERIALIZABLE = 1 << 0;
|
public static final int FLAG_SERIALIZABLE = 1 << 0;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Flag for alternate metafactories indicating the lambda object implements other marker interfaces
|
* Flag for alternate metafactories indicating the lambda object implements
|
||||||
|
* other marker interfaces
|
||||||
* besides Serializable
|
* besides Serializable
|
||||||
*/
|
*/
|
||||||
public static final int FLAG_MARKERS = 1 << 1;
|
public static final int FLAG_MARKERS = 1 << 1;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Flag for alternate metafactories indicating the lambda object requires
|
||||||
|
* additional bridge methods
|
||||||
|
*/
|
||||||
|
public static final int FLAG_BRIDGES = 1 << 2;
|
||||||
|
|
||||||
private static final Class<?>[] EMPTY_CLASS_ARRAY = new Class<?>[0];
|
private static final Class<?>[] EMPTY_CLASS_ARRAY = new Class<?>[0];
|
||||||
|
private static final MethodType[] EMPTY_MT_ARRAY = new MethodType[0];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Standard meta-factory for conversion of lambda expressions or method references to functional interfaces.
|
* Standard meta-factory for conversion of lambda expressions or method
|
||||||
|
* references to functional interfaces.
|
||||||
*
|
*
|
||||||
* @param caller Stacked automatically by VM; represents a lookup context with the accessibility privileges
|
* @param caller Stacked automatically by VM; represents a lookup context
|
||||||
* of the caller.
|
* with the accessibility privileges of the caller.
|
||||||
* @param invokedName Stacked automatically by VM; the name of the invoked method as it appears at the call site.
|
* @param invokedName Stacked automatically by VM; the name of the invoked
|
||||||
* Currently unused.
|
* method as it appears at the call site.
|
||||||
* @param invokedType Stacked automatically by VM; the signature of the invoked method, which includes the
|
* Used as the name of the functional interface method
|
||||||
* expected static type of the returned lambda object, and the static types of the captured
|
* to which the lambda or method reference is being
|
||||||
* arguments for the lambda. In the event that the implementation method is an instance method,
|
* converted.
|
||||||
* the first argument in the invocation signature will correspond to the receiver.
|
* @param invokedType Stacked automatically by VM; the signature of the
|
||||||
* @param samMethod The primary method in the functional interface to which the lambda or method reference is
|
* invoked method, which includes the expected static
|
||||||
* being converted, represented as a method handle.
|
* type of the returned lambda object, and the static
|
||||||
* @param implMethod The implementation method which should be called (with suitable adaptation of argument
|
* types of the captured arguments for the lambda.
|
||||||
* types, return types, and adjustment for captured arguments) when methods of the resulting
|
* In the event that the implementation method is an
|
||||||
* functional interface instance are invoked.
|
* instance method, the first argument in the invocation
|
||||||
* @param instantiatedMethodType The signature of the primary functional interface method after type variables
|
* signature will correspond to the receiver.
|
||||||
* are substituted with their instantiation from the capture site
|
* @param samMethodType MethodType of the method in the functional interface
|
||||||
* @return a CallSite, which, when invoked, will return an instance of the functional interface
|
* to which the lambda or method reference is being
|
||||||
* @throws ReflectiveOperationException if the caller is not able to reconstruct one of the method handles
|
* converted, represented as a MethodType.
|
||||||
* @throws LambdaConversionException If any of the meta-factory protocol invariants are violated
|
* @param implMethod The implementation method which should be called
|
||||||
|
* (with suitable adaptation of argument types, return
|
||||||
|
* types, and adjustment for captured arguments) when
|
||||||
|
* methods of the resulting functional interface instance
|
||||||
|
* are invoked.
|
||||||
|
* @param instantiatedMethodType The signature of the primary functional
|
||||||
|
* interface method after type variables
|
||||||
|
* are substituted with their instantiation
|
||||||
|
* from the capture site
|
||||||
|
* @return a CallSite, which, when invoked, will return an instance of the
|
||||||
|
* functional interface
|
||||||
|
* @throws ReflectiveOperationException if the caller is not able to
|
||||||
|
* reconstruct one of the method handles
|
||||||
|
* @throws LambdaConversionException If any of the meta-factory protocol
|
||||||
|
* invariants are violated
|
||||||
*/
|
*/
|
||||||
public static CallSite metaFactory(MethodHandles.Lookup caller,
|
public static CallSite metafactory(MethodHandles.Lookup caller,
|
||||||
String invokedName,
|
String invokedName,
|
||||||
MethodType invokedType,
|
MethodType invokedType,
|
||||||
MethodHandle samMethod,
|
MethodType samMethodType,
|
||||||
MethodHandle implMethod,
|
MethodHandle implMethod,
|
||||||
MethodType instantiatedMethodType)
|
MethodType instantiatedMethodType)
|
||||||
throws ReflectiveOperationException, LambdaConversionException {
|
throws ReflectiveOperationException, LambdaConversionException {
|
||||||
AbstractValidatingLambdaMetafactory mf;
|
AbstractValidatingLambdaMetafactory mf;
|
||||||
mf = new InnerClassLambdaMetafactory(caller, invokedType, samMethod, implMethod, instantiatedMethodType,
|
mf = new InnerClassLambdaMetafactory(caller, invokedType,
|
||||||
0, EMPTY_CLASS_ARRAY);
|
invokedName, samMethodType,
|
||||||
|
implMethod, instantiatedMethodType,
|
||||||
|
false, EMPTY_CLASS_ARRAY, EMPTY_MT_ARRAY);
|
||||||
mf.validateMetafactoryArgs();
|
mf.validateMetafactoryArgs();
|
||||||
return mf.buildCallSite();
|
return mf.buildCallSite();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Alternate meta-factory for conversion of lambda expressions or method references to functional interfaces,
|
* Alternate meta-factory for conversion of lambda expressions or method
|
||||||
* which supports serialization and other uncommon options.
|
* references to functional interfaces, which supports serialization and
|
||||||
|
* other uncommon options.
|
||||||
*
|
*
|
||||||
* The declared argument list for this method is:
|
* The declared argument list for this method is:
|
||||||
*
|
*
|
||||||
* CallSite altMetaFactory(MethodHandles.Lookup caller,
|
* CallSite altMetafactory(MethodHandles.Lookup caller,
|
||||||
* String invokedName,
|
* String invokedName,
|
||||||
* MethodType invokedType,
|
* MethodType invokedType,
|
||||||
* Object... args)
|
* Object... args)
|
||||||
*
|
*
|
||||||
* but it behaves as if the argument list is:
|
* but it behaves as if the argument list is:
|
||||||
*
|
*
|
||||||
* CallSite altMetaFactory(MethodHandles.Lookup caller,
|
* CallSite altMetafactory(MethodHandles.Lookup caller,
|
||||||
* String invokedName,
|
* String invokedName,
|
||||||
* MethodType invokedType,
|
* MethodType invokedType,
|
||||||
* MethodHandle samMethod
|
* MethodType samMethodType
|
||||||
* MethodHandle implMethod,
|
* MethodHandle implMethod,
|
||||||
* MethodType instantiatedMethodType,
|
* MethodType instantiatedMethodType,
|
||||||
* int flags,
|
* int flags,
|
||||||
* int markerInterfaceCount, // IF flags has MARKERS set
|
* int markerInterfaceCount, // IF flags has MARKERS set
|
||||||
* Class... markerInterfaces // IF flags has MARKERS set
|
* Class... markerInterfaces // IF flags has MARKERS set
|
||||||
|
* int bridgeCount, // IF flags has BRIDGES set
|
||||||
|
* MethodType... bridges // IF flags has BRIDGES set
|
||||||
* )
|
* )
|
||||||
*
|
*
|
||||||
*
|
*
|
||||||
* @param caller Stacked automatically by VM; represents a lookup context with the accessibility privileges
|
* @param caller Stacked automatically by VM; represents a lookup context
|
||||||
* of the caller.
|
* with the accessibility privileges of the caller.
|
||||||
* @param invokedName Stacked automatically by VM; the name of the invoked method as it appears at the call site.
|
* @param invokedName Stacked automatically by VM; the name of the invoked
|
||||||
* Currently unused.
|
* method as it appears at the call site.
|
||||||
* @param invokedType Stacked automatically by VM; the signature of the invoked method, which includes thefu
|
* Used as the name of the functional interface method
|
||||||
* expected static type of the returned lambda object, and the static types of the captured
|
* to which the lambda or method reference is being
|
||||||
* arguments for the lambda. In the event that the implementation method is an instance method,
|
* converted.
|
||||||
* the first argument in the invocation signature will correspond to the receiver.
|
* @param invokedType Stacked automatically by VM; the signature of the
|
||||||
* @param args argument to pass, flags, marker interface count, and marker interfaces as described above
|
* invoked method, which includes the expected static
|
||||||
* @return a CallSite, which, when invoked, will return an instance of the functional interface
|
* type of the returned lambda object, and the static
|
||||||
* @throws ReflectiveOperationException if the caller is not able to reconstruct one of the method handles
|
* types of the captured arguments for the lambda.
|
||||||
* @throws LambdaConversionException If any of the meta-factory protocol invariants are violated
|
* In the event that the implementation method is an
|
||||||
|
* instance method, the first argument in the invocation
|
||||||
|
* signature will correspond to the receiver.
|
||||||
|
* @param args flags and optional arguments, as described above
|
||||||
|
* @return a CallSite, which, when invoked, will return an instance of the
|
||||||
|
* functional interface
|
||||||
|
* @throws ReflectiveOperationException if the caller is not able to
|
||||||
|
* reconstruct one of the method handles
|
||||||
|
* @throws LambdaConversionException If any of the meta-factory protocol
|
||||||
|
* invariants are violated
|
||||||
*/
|
*/
|
||||||
public static CallSite altMetaFactory(MethodHandles.Lookup caller,
|
public static CallSite altMetafactory(MethodHandles.Lookup caller,
|
||||||
String invokedName,
|
String invokedName,
|
||||||
MethodType invokedType,
|
MethodType invokedType,
|
||||||
Object... args)
|
Object... args)
|
||||||
throws ReflectiveOperationException, LambdaConversionException {
|
throws ReflectiveOperationException, LambdaConversionException {
|
||||||
MethodHandle samMethod = (MethodHandle)args[0];
|
MethodType samMethodType = (MethodType)args[0];
|
||||||
MethodHandle implMethod = (MethodHandle)args[1];
|
MethodHandle implMethod = (MethodHandle)args[1];
|
||||||
MethodType instantiatedMethodType = (MethodType)args[2];
|
MethodType instantiatedMethodType = (MethodType)args[2];
|
||||||
int flags = (Integer) args[3];
|
int flags = (Integer) args[3];
|
||||||
Class<?>[] markerInterfaces;
|
Class<?>[] markerInterfaces;
|
||||||
|
MethodType[] bridges;
|
||||||
int argIndex = 4;
|
int argIndex = 4;
|
||||||
if ((flags & FLAG_MARKERS) != 0) {
|
if ((flags & FLAG_MARKERS) != 0) {
|
||||||
int markerCount = (Integer) args[argIndex++];
|
int markerCount = (Integer) args[argIndex++];
|
||||||
@ -248,9 +285,33 @@ public class LambdaMetafactory {
|
|||||||
}
|
}
|
||||||
else
|
else
|
||||||
markerInterfaces = EMPTY_CLASS_ARRAY;
|
markerInterfaces = EMPTY_CLASS_ARRAY;
|
||||||
AbstractValidatingLambdaMetafactory mf;
|
if ((flags & FLAG_BRIDGES) != 0) {
|
||||||
mf = new InnerClassLambdaMetafactory(caller, invokedType, samMethod, implMethod, instantiatedMethodType,
|
int bridgeCount = (Integer) args[argIndex++];
|
||||||
flags, markerInterfaces);
|
bridges = new MethodType[bridgeCount];
|
||||||
|
System.arraycopy(args, argIndex, bridges, 0, bridgeCount);
|
||||||
|
argIndex += bridgeCount;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
bridges = EMPTY_MT_ARRAY;
|
||||||
|
|
||||||
|
boolean foundSerializableSupertype = Serializable.class.isAssignableFrom(invokedType.returnType());
|
||||||
|
for (Class<?> c : markerInterfaces)
|
||||||
|
foundSerializableSupertype |= Serializable.class.isAssignableFrom(c);
|
||||||
|
boolean isSerializable = ((flags & LambdaMetafactory.FLAG_SERIALIZABLE) != 0)
|
||||||
|
|| foundSerializableSupertype;
|
||||||
|
|
||||||
|
if (isSerializable && !foundSerializableSupertype) {
|
||||||
|
markerInterfaces = Arrays.copyOf(markerInterfaces, markerInterfaces.length + 1);
|
||||||
|
markerInterfaces[markerInterfaces.length-1] = Serializable.class;
|
||||||
|
}
|
||||||
|
|
||||||
|
AbstractValidatingLambdaMetafactory mf
|
||||||
|
= new InnerClassLambdaMetafactory(caller, invokedType,
|
||||||
|
invokedName, samMethodType,
|
||||||
|
implMethod,
|
||||||
|
instantiatedMethodType,
|
||||||
|
isSerializable,
|
||||||
|
markerInterfaces, bridges);
|
||||||
mf.validateMetafactoryArgs();
|
mf.validateMetafactoryArgs();
|
||||||
return mf.buildCallSite();
|
return mf.buildCallSite();
|
||||||
}
|
}
|
||||||
|
@ -44,7 +44,6 @@ public final class SerializedLambda implements Serializable {
|
|||||||
private final String functionalInterfaceClass;
|
private final String functionalInterfaceClass;
|
||||||
private final String functionalInterfaceMethodName;
|
private final String functionalInterfaceMethodName;
|
||||||
private final String functionalInterfaceMethodSignature;
|
private final String functionalInterfaceMethodSignature;
|
||||||
private final int functionalInterfaceMethodKind;
|
|
||||||
private final String implClass;
|
private final String implClass;
|
||||||
private final String implMethodName;
|
private final String implMethodName;
|
||||||
private final String implMethodSignature;
|
private final String implMethodSignature;
|
||||||
@ -53,28 +52,32 @@ public final class SerializedLambda implements Serializable {
|
|||||||
private final Object[] capturedArgs;
|
private final Object[] capturedArgs;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a {@code SerializedLambda} from the low-level information present at the lambda factory site.
|
* Create a {@code SerializedLambda} from the low-level information present
|
||||||
|
* at the lambda factory site.
|
||||||
*
|
*
|
||||||
* @param capturingClass The class in which the lambda expression appears
|
* @param capturingClass The class in which the lambda expression appears
|
||||||
* @param functionalInterfaceMethodKind Method handle kind (see {@link MethodHandleInfo}) for the
|
* @param functionalInterfaceClass Name, in slash-delimited form, of static
|
||||||
* functional interface method handle present at the lambda factory site
|
* type of the returned lambda object
|
||||||
* @param functionalInterfaceClass Name, in slash-delimited form, for the functional interface class present at the
|
* @param functionalInterfaceMethodName Name of the functional interface
|
||||||
|
* method for the present at the
|
||||||
* lambda factory site
|
* lambda factory site
|
||||||
* @param functionalInterfaceMethodName Name of the primary method for the functional interface present at the
|
* @param functionalInterfaceMethodSignature Signature of the functional
|
||||||
* lambda factory site
|
* interface method present at
|
||||||
* @param functionalInterfaceMethodSignature Signature of the primary method for the functional interface present
|
* the lambda factory site
|
||||||
* at the lambda factory site
|
|
||||||
* @param implMethodKind Method handle kind for the implementation method
|
* @param implMethodKind Method handle kind for the implementation method
|
||||||
* @param implClass Name, in slash-delimited form, for the class holding the implementation method
|
* @param implClass Name, in slash-delimited form, for the class holding
|
||||||
|
* the implementation method
|
||||||
* @param implMethodName Name of the implementation method
|
* @param implMethodName Name of the implementation method
|
||||||
* @param implMethodSignature Signature of the implementation method
|
* @param implMethodSignature Signature of the implementation method
|
||||||
* @param instantiatedMethodType The signature of the primary functional interface method after type variables
|
* @param instantiatedMethodType The signature of the primary functional
|
||||||
* are substituted with their instantiation from the capture site
|
* interface method after type variables
|
||||||
* @param capturedArgs The dynamic arguments to the lambda factory site, which represent variables captured by
|
* are substituted with their instantiation
|
||||||
|
* from the capture site
|
||||||
|
* @param capturedArgs The dynamic arguments to the lambda factory site,
|
||||||
|
* which represent variables captured by
|
||||||
* the lambda
|
* the lambda
|
||||||
*/
|
*/
|
||||||
public SerializedLambda(Class<?> capturingClass,
|
public SerializedLambda(Class<?> capturingClass,
|
||||||
int functionalInterfaceMethodKind,
|
|
||||||
String functionalInterfaceClass,
|
String functionalInterfaceClass,
|
||||||
String functionalInterfaceMethodName,
|
String functionalInterfaceMethodName,
|
||||||
String functionalInterfaceMethodSignature,
|
String functionalInterfaceMethodSignature,
|
||||||
@ -85,7 +88,6 @@ public final class SerializedLambda implements Serializable {
|
|||||||
String instantiatedMethodType,
|
String instantiatedMethodType,
|
||||||
Object[] capturedArgs) {
|
Object[] capturedArgs) {
|
||||||
this.capturingClass = capturingClass;
|
this.capturingClass = capturingClass;
|
||||||
this.functionalInterfaceMethodKind = functionalInterfaceMethodKind;
|
|
||||||
this.functionalInterfaceClass = functionalInterfaceClass;
|
this.functionalInterfaceClass = functionalInterfaceClass;
|
||||||
this.functionalInterfaceMethodName = functionalInterfaceMethodName;
|
this.functionalInterfaceMethodName = functionalInterfaceMethodName;
|
||||||
this.functionalInterfaceMethodSignature = functionalInterfaceMethodSignature;
|
this.functionalInterfaceMethodSignature = functionalInterfaceMethodSignature;
|
||||||
@ -106,10 +108,10 @@ public final class SerializedLambda implements Serializable {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the name of the functional interface class to which this
|
* Get the name of the invoked type to which this
|
||||||
* lambda has been converted
|
* lambda has been converted
|
||||||
* @return the name of the functional interface this lambda has
|
* @return the name of the functional interface class to which
|
||||||
* been converted to
|
* this lambda has been converted
|
||||||
*/
|
*/
|
||||||
public String getFunctionalInterfaceClass() {
|
public String getFunctionalInterfaceClass() {
|
||||||
return functionalInterfaceClass;
|
return functionalInterfaceClass;
|
||||||
@ -134,17 +136,6 @@ public final class SerializedLambda implements Serializable {
|
|||||||
return functionalInterfaceMethodSignature;
|
return functionalInterfaceMethodSignature;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Get the method handle kind (see {@link MethodHandleInfo}) of
|
|
||||||
* the primary method for the functional interface to which this
|
|
||||||
* lambda has been converted
|
|
||||||
* @return the method handle kind of the primary method of
|
|
||||||
* functional interface
|
|
||||||
*/
|
|
||||||
public int getFunctionalInterfaceMethodKind() {
|
|
||||||
return functionalInterfaceMethodKind;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Get the name of the class containing the implementation
|
* Get the name of the class containing the implementation
|
||||||
* method.
|
* method.
|
||||||
@ -234,11 +225,17 @@ public final class SerializedLambda implements Serializable {
|
|||||||
|
|
||||||
@Override
|
@Override
|
||||||
public String toString() {
|
public String toString() {
|
||||||
return String.format("SerializedLambda[capturingClass=%s, functionalInterfaceMethod=%s %s.%s:%s, " +
|
String implKind=MethodHandleInfo.getReferenceKindString(implMethodKind);
|
||||||
"implementation=%s %s.%s:%s, instantiatedMethodType=%s, numCaptured=%d]",
|
return String.format("SerializedLambda[%s=%s, %s=%s.%s:%s, " +
|
||||||
capturingClass, MethodHandleInfo.getReferenceKindString(functionalInterfaceMethodKind),
|
"%s=%s %s.%s:%s, %s=%s, %s=%d]",
|
||||||
functionalInterfaceClass, functionalInterfaceMethodName, functionalInterfaceMethodSignature,
|
"capturingClass", capturingClass,
|
||||||
MethodHandleInfo.getReferenceKindString(implMethodKind), implClass, implMethodName,
|
"functionalInterfaceMethod", functionalInterfaceClass,
|
||||||
implMethodSignature, instantiatedMethodType, capturedArgs.length);
|
functionalInterfaceMethodName,
|
||||||
|
functionalInterfaceMethodSignature,
|
||||||
|
"implementation",
|
||||||
|
implKind,
|
||||||
|
implClass, implMethodName, implMethodSignature,
|
||||||
|
"instantiatedMethodType", instantiatedMethodType,
|
||||||
|
"numCaptured", capturedArgs.length);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright 2013 Sun Microsystems, Inc. All Rights Reserved.
|
* Copyright (c) 2013, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -16,10 +16,10 @@
|
|||||||
* 2 along with this work; if not, write to the Free Software Foundation,
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
||||||
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
*
|
*
|
||||||
* Please contact Sun Microsystems, Inc., 4150 Network Circle, Santa Clara,
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||||||
* CA 95054 USA or visit www.sun.com if you need additional information or
|
* or visit www.oracle.com if you need additional information or have any
|
||||||
* have any questions.
|
* questions.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* @test
|
* @test
|
||||||
|
Loading…
Reference in New Issue
Block a user