From 5f8b5ceaee176e6391416197ed8020159814fcb9 Mon Sep 17 00:00:00 2001 From: Lois Foltan Date: Fri, 23 Jan 2015 09:54:33 -0500 Subject: [PATCH 01/58] 8067480: Crash in klassItable::initialize_itable_for_interface when running vm.runtime.defmeth.StaticMethodsTest A static method should be skipped in entirety by find_instance_method(), searching should continue to find an overpass method of the same name and signature if present. Reviewed-by: acorn, hseigel --- hotspot/src/share/vm/classfile/verifier.cpp | 4 +- .../src/share/vm/interpreter/linkResolver.cpp | 9 +-- hotspot/src/share/vm/oops/arrayKlass.cpp | 9 ++- hotspot/src/share/vm/oops/arrayKlass.hpp | 4 +- hotspot/src/share/vm/oops/instanceKlass.cpp | 60 ++++++++++++------- hotspot/src/share/vm/oops/instanceKlass.hpp | 15 +++-- hotspot/src/share/vm/oops/klass.cpp | 4 +- hotspot/src/share/vm/oops/klass.hpp | 10 ++-- hotspot/src/share/vm/oops/klassVtable.cpp | 6 +- hotspot/src/share/vm/prims/jvm.cpp | 4 +- hotspot/src/share/vm/prims/nativeLookup.cpp | 4 +- 11 files changed, 76 insertions(+), 53 deletions(-) diff --git a/hotspot/src/share/vm/classfile/verifier.cpp b/hotspot/src/share/vm/classfile/verifier.cpp index d08a818ae65..d42492b4ed0 100644 --- a/hotspot/src/share/vm/classfile/verifier.cpp +++ b/hotspot/src/share/vm/classfile/verifier.cpp @@ -1950,7 +1950,7 @@ bool ClassVerifier::is_protected_access(instanceKlassHandle this_class, InstanceKlass* target_instance = InstanceKlass::cast(target_class); fieldDescriptor fd; if (is_method) { - Method* m = target_instance->uncached_lookup_method(field_name, field_sig, Klass::normal); + Method* m = target_instance->uncached_lookup_method(field_name, field_sig, Klass::find_overpass); if (m != NULL && m->is_protected()) { if (!this_class->is_same_class_package(m->method_holder())) { return true; @@ -2487,7 +2487,7 @@ void ClassVerifier::verify_invoke_init( Method* m = InstanceKlass::cast(ref_klass)->uncached_lookup_method( vmSymbols::object_initializer_name(), cp->signature_ref_at(bcs->get_index_u2()), - Klass::normal); + Klass::find_overpass); // Do nothing if method is not found. Let resolution detect the error. if (m != NULL) { instanceKlassHandle mh(THREAD, m->method_holder()); diff --git a/hotspot/src/share/vm/interpreter/linkResolver.cpp b/hotspot/src/share/vm/interpreter/linkResolver.cpp index 5bf368a1765..561b3352bd4 100644 --- a/hotspot/src/share/vm/interpreter/linkResolver.cpp +++ b/hotspot/src/share/vm/interpreter/linkResolver.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -289,11 +289,11 @@ void LinkResolver::lookup_method_in_klasses(methodHandle& result, KlassHandle kl // returns first instance method // Looks up method in classes, then looks up local default methods void LinkResolver::lookup_instance_method_in_klasses(methodHandle& result, KlassHandle klass, Symbol* name, Symbol* signature, TRAPS) { - Method* result_oop = klass->uncached_lookup_method(name, signature, Klass::normal); + Method* result_oop = klass->uncached_lookup_method(name, signature, Klass::find_overpass); result = methodHandle(THREAD, result_oop); while (!result.is_null() && result->is_static() && result->method_holder()->super() != NULL) { KlassHandle super_klass = KlassHandle(THREAD, result->method_holder()->super()); - result = methodHandle(THREAD, super_klass->uncached_lookup_method(name, signature, Klass::normal)); + result = methodHandle(THREAD, super_klass->uncached_lookup_method(name, signature, Klass::find_overpass)); } if (klass->oop_is_array()) { @@ -320,7 +320,8 @@ int LinkResolver::vtable_index_of_interface_method(KlassHandle klass, // First check in default method array if (!resolved_method->is_abstract() && (InstanceKlass::cast(klass())->default_methods() != NULL)) { - int index = InstanceKlass::find_method_index(InstanceKlass::cast(klass())->default_methods(), name, signature, false); + int index = InstanceKlass::find_method_index(InstanceKlass::cast(klass())->default_methods(), + name, signature, Klass::find_overpass, Klass::find_static); if (index >= 0 ) { vtable_index = InstanceKlass::cast(klass())->default_vtable_indices()->at(index); } diff --git a/hotspot/src/share/vm/oops/arrayKlass.cpp b/hotspot/src/share/vm/oops/arrayKlass.cpp index 740ae166adf..74898de3f07 100644 --- a/hotspot/src/share/vm/oops/arrayKlass.cpp +++ b/hotspot/src/share/vm/oops/arrayKlass.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -71,10 +71,13 @@ Klass* ArrayKlass::find_field(Symbol* name, Symbol* sig, fieldDescriptor* fd) co return super()->find_field(name, sig, fd); } -Method* ArrayKlass::uncached_lookup_method(Symbol* name, Symbol* signature, MethodLookupMode mode) const { +Method* ArrayKlass::uncached_lookup_method(Symbol* name, Symbol* signature, OverpassLookupMode overpass_mode) const { // There are no methods in an array klass but the super class (Object) has some assert(super(), "super klass must be present"); - return super()->uncached_lookup_method(name, signature, mode); + // Always ignore overpass methods in superclasses, although technically the + // super klass of an array, (j.l.Object) should not have + // any overpass methods present. + return super()->uncached_lookup_method(name, signature, Klass::skip_overpass); } ArrayKlass::ArrayKlass(Symbol* name) { diff --git a/hotspot/src/share/vm/oops/arrayKlass.hpp b/hotspot/src/share/vm/oops/arrayKlass.hpp index 248fde0373a..3ec39b2682b 100644 --- a/hotspot/src/share/vm/oops/arrayKlass.hpp +++ b/hotspot/src/share/vm/oops/arrayKlass.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -82,7 +82,7 @@ class ArrayKlass: public Klass { Klass* find_field(Symbol* name, Symbol* sig, fieldDescriptor* fd) const; // Lookup operations - Method* uncached_lookup_method(Symbol* name, Symbol* signature, MethodLookupMode mode) const; + Method* uncached_lookup_method(Symbol* name, Symbol* signature, OverpassLookupMode overpass_mode) const; // Casting from Klass* static ArrayKlass* cast(Klass* k) { diff --git a/hotspot/src/share/vm/oops/instanceKlass.cpp b/hotspot/src/share/vm/oops/instanceKlass.cpp index cf663ea72d1..1cf7619f57f 100644 --- a/hotspot/src/share/vm/oops/instanceKlass.cpp +++ b/hotspot/src/share/vm/oops/instanceKlass.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -1416,33 +1416,33 @@ static int binary_search(Array* methods, Symbol* name) { // find_method looks up the name/signature in the local methods array Method* InstanceKlass::find_method(Symbol* name, Symbol* signature) const { - return find_method_impl(name, signature, false); + return find_method_impl(name, signature, find_overpass, find_static); } -Method* InstanceKlass::find_method_impl(Symbol* name, Symbol* signature, bool skipping_overpass) const { - return InstanceKlass::find_method_impl(methods(), name, signature, skipping_overpass); +Method* InstanceKlass::find_method_impl(Symbol* name, Symbol* signature, + OverpassLookupMode overpass_mode, StaticLookupMode static_mode) const { + return InstanceKlass::find_method_impl(methods(), name, signature, overpass_mode, static_mode); } // find_instance_method looks up the name/signature in the local methods array // and skips over static methods Method* InstanceKlass::find_instance_method( Array* methods, Symbol* name, Symbol* signature) { - Method* meth = InstanceKlass::find_method(methods, name, signature); - if (meth != NULL && meth->is_static()) { - meth = NULL; - } + Method* meth = InstanceKlass::find_method_impl(methods, name, signature, + find_overpass, skip_static); + assert(((meth == NULL) || !meth->is_static()), "find_instance_method should have skipped statics"); return meth; } // find_method looks up the name/signature in the local methods array Method* InstanceKlass::find_method( Array* methods, Symbol* name, Symbol* signature) { - return InstanceKlass::find_method_impl(methods, name, signature, false); + return InstanceKlass::find_method_impl(methods, name, signature, find_overpass, find_static); } Method* InstanceKlass::find_method_impl( - Array* methods, Symbol* name, Symbol* signature, bool skipping_overpass) { - int hit = find_method_index(methods, name, signature, skipping_overpass); + Array* methods, Symbol* name, Symbol* signature, OverpassLookupMode overpass_mode, StaticLookupMode static_mode) { + int hit = find_method_index(methods, name, signature, overpass_mode, static_mode); return hit >= 0 ? methods->at(hit): NULL; } @@ -1454,13 +1454,19 @@ Method* InstanceKlass::find_method_impl( // is important during method resolution to prefer a static method, for example, // over an overpass method. int InstanceKlass::find_method_index( - Array* methods, Symbol* name, Symbol* signature, bool skipping_overpass) { + Array* methods, Symbol* name, Symbol* signature, OverpassLookupMode overpass_mode, StaticLookupMode static_mode) { + bool skipping_overpass = (overpass_mode == skip_overpass); + bool skipping_static = (static_mode == skip_static); int hit = binary_search(methods, name); if (hit != -1) { Method* m = methods->at(hit); // Do linear search to find matching signature. First, quick check // for common case, ignoring overpasses if requested. - if ((m->signature() == signature) && (!skipping_overpass || !m->is_overpass())) return hit; + if ((m->signature() == signature) && + (!skipping_overpass || !m->is_overpass()) && + (!skipping_static || !m->is_static())) { + return hit; + } // search downwards through overloaded methods int i; @@ -1468,18 +1474,26 @@ int InstanceKlass::find_method_index( Method* m = methods->at(i); assert(m->is_method(), "must be method"); if (m->name() != name) break; - if ((m->signature() == signature) && (!skipping_overpass || !m->is_overpass())) return i; + if ((m->signature() == signature) && + (!skipping_overpass || !m->is_overpass()) && + (!skipping_static || !m->is_static())) { + return i; + } } // search upwards for (i = hit + 1; i < methods->length(); ++i) { Method* m = methods->at(i); assert(m->is_method(), "must be method"); if (m->name() != name) break; - if ((m->signature() == signature) && (!skipping_overpass || !m->is_overpass())) return i; + if ((m->signature() == signature) && + (!skipping_overpass || !m->is_overpass()) && + (!skipping_static || !m->is_static())) { + return i; + } } // not found #ifdef ASSERT - int index = skipping_overpass ? -1 : linear_search(methods, name, signature); + int index = (skipping_overpass || skipping_static) ? -1 : linear_search(methods, name, signature); assert(index == -1, err_msg("binary search should have found entry %d", index)); #endif } @@ -1505,16 +1519,16 @@ int InstanceKlass::find_method_by_name( // uncached_lookup_method searches both the local class methods array and all // superclasses methods arrays, skipping any overpass methods in superclasses. -Method* InstanceKlass::uncached_lookup_method(Symbol* name, Symbol* signature, MethodLookupMode mode) const { - MethodLookupMode lookup_mode = mode; +Method* InstanceKlass::uncached_lookup_method(Symbol* name, Symbol* signature, OverpassLookupMode overpass_mode) const { + OverpassLookupMode overpass_local_mode = overpass_mode; Klass* klass = const_cast(this); while (klass != NULL) { - Method* method = InstanceKlass::cast(klass)->find_method_impl(name, signature, (lookup_mode == skip_overpass)); + Method* method = InstanceKlass::cast(klass)->find_method_impl(name, signature, overpass_local_mode, find_static); if (method != NULL) { return method; } klass = InstanceKlass::cast(klass)->super(); - lookup_mode = skip_overpass; // Always ignore overpass methods in superclasses + overpass_local_mode = skip_overpass; // Always ignore overpass methods in superclasses } return NULL; } @@ -1544,7 +1558,7 @@ Method* InstanceKlass::lookup_method_in_ordered_interfaces(Symbol* name, } // Look up interfaces if (m == NULL) { - m = lookup_method_in_all_interfaces(name, signature, normal); + m = lookup_method_in_all_interfaces(name, signature, find_defaults); } return m; } @@ -1554,7 +1568,7 @@ Method* InstanceKlass::lookup_method_in_ordered_interfaces(Symbol* name, // They should only be found in the initial InterfaceMethodRef Method* InstanceKlass::lookup_method_in_all_interfaces(Symbol* name, Symbol* signature, - MethodLookupMode mode) const { + DefaultsLookupMode defaults_mode) const { Array* all_ifs = transitive_interfaces(); int num_ifs = all_ifs->length(); InstanceKlass *ik = NULL; @@ -1562,7 +1576,7 @@ Method* InstanceKlass::lookup_method_in_all_interfaces(Symbol* name, ik = InstanceKlass::cast(all_ifs->at(i)); Method* m = ik->lookup_method(name, signature); if (m != NULL && m->is_public() && !m->is_static() && - ((mode != skip_defaults) || !m->is_default_method())) { + ((defaults_mode != skip_defaults) || !m->is_default_method())) { return m; } } diff --git a/hotspot/src/share/vm/oops/instanceKlass.hpp b/hotspot/src/share/vm/oops/instanceKlass.hpp index 94a3c80ea98..6bca02b5202 100644 --- a/hotspot/src/share/vm/oops/instanceKlass.hpp +++ b/hotspot/src/share/vm/oops/instanceKlass.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -493,14 +493,15 @@ class InstanceKlass: public Klass { static Method* find_instance_method(Array* methods, Symbol* name, Symbol* signature); // find a local method index in default_methods (returns -1 if not found) - static int find_method_index(Array* methods, Symbol* name, Symbol* signature, bool skipping_overpass); + static int find_method_index(Array* methods, Symbol* name, Symbol* signature, + OverpassLookupMode overpass_mode, StaticLookupMode static_mode); // lookup operation (returns NULL if not found) - Method* uncached_lookup_method(Symbol* name, Symbol* signature, MethodLookupMode mode) const; + Method* uncached_lookup_method(Symbol* name, Symbol* signature, OverpassLookupMode overpass_mode) const; // lookup a method in all the interfaces that this class implements // (returns NULL if not found) - Method* lookup_method_in_all_interfaces(Symbol* name, Symbol* signature, MethodLookupMode mode) const; + Method* lookup_method_in_all_interfaces(Symbol* name, Symbol* signature, DefaultsLookupMode defaults_mode) const; // lookup a method in local defaults then in all interfaces // (returns NULL if not found) @@ -1052,8 +1053,10 @@ private: Klass* array_klass_impl(bool or_null, TRAPS); // find a local method (returns NULL if not found) - Method* find_method_impl(Symbol* name, Symbol* signature, bool skipping_overpass) const; - static Method* find_method_impl(Array* methods, Symbol* name, Symbol* signature, bool skipping_overpass); + Method* find_method_impl(Symbol* name, Symbol* signature, + OverpassLookupMode overpass_mode, StaticLookupMode static_mode) const; + static Method* find_method_impl(Array* methods, Symbol* name, Symbol* signature, + OverpassLookupMode overpass_mode, StaticLookupMode static_mode); // Free CHeap allocated fields. void release_C_heap_structures(); diff --git a/hotspot/src/share/vm/oops/klass.cpp b/hotspot/src/share/vm/oops/klass.cpp index 56de60fa37b..0ae07185781 100644 --- a/hotspot/src/share/vm/oops/klass.cpp +++ b/hotspot/src/share/vm/oops/klass.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -140,7 +140,7 @@ Klass* Klass::find_field(Symbol* name, Symbol* sig, fieldDescriptor* fd) const { return NULL; } -Method* Klass::uncached_lookup_method(Symbol* name, Symbol* signature, MethodLookupMode mode) const { +Method* Klass::uncached_lookup_method(Symbol* name, Symbol* signature, OverpassLookupMode overpass_mode) const { #ifdef ASSERT tty->print_cr("Error: uncached_lookup_method called on a klass oop." " Likely error: reflection method does not correctly" diff --git a/hotspot/src/share/vm/oops/klass.hpp b/hotspot/src/share/vm/oops/klass.hpp index fd7691462f3..f1b6a72ad7d 100644 --- a/hotspot/src/share/vm/oops/klass.hpp +++ b/hotspot/src/share/vm/oops/klass.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -164,7 +164,9 @@ protected: void* operator new(size_t size, ClassLoaderData* loader_data, size_t word_size, TRAPS) throw(); public: - enum MethodLookupMode { normal, skip_overpass, skip_defaults }; + enum DefaultsLookupMode { find_defaults, skip_defaults }; + enum OverpassLookupMode { find_overpass, skip_overpass }; + enum StaticLookupMode { find_static, skip_static }; bool is_klass() const volatile { return true; } @@ -413,10 +415,10 @@ protected: // lookup operation for MethodLookupCache friend class MethodLookupCache; virtual Klass* find_field(Symbol* name, Symbol* signature, fieldDescriptor* fd) const; - virtual Method* uncached_lookup_method(Symbol* name, Symbol* signature, MethodLookupMode mode) const; + virtual Method* uncached_lookup_method(Symbol* name, Symbol* signature, OverpassLookupMode overpass_mode) const; public: Method* lookup_method(Symbol* name, Symbol* signature) const { - return uncached_lookup_method(name, signature, normal); + return uncached_lookup_method(name, signature, find_overpass); } // array class with specific rank diff --git a/hotspot/src/share/vm/oops/klassVtable.cpp b/hotspot/src/share/vm/oops/klassVtable.cpp index 6978247d318..a50b77ce27d 100644 --- a/hotspot/src/share/vm/oops/klassVtable.cpp +++ b/hotspot/src/share/vm/oops/klassVtable.cpp @@ -649,7 +649,7 @@ bool klassVtable::needs_new_vtable_entry(methodHandle target_method, // this check for all access permissions. InstanceKlass *sk = InstanceKlass::cast(super); if (sk->has_miranda_methods()) { - if (sk->lookup_method_in_all_interfaces(name, signature, Klass::normal) != NULL) { + if (sk->lookup_method_in_all_interfaces(name, signature, Klass::find_defaults) != NULL) { return false; // found a matching miranda; we do not need a new entry } } @@ -725,7 +725,7 @@ bool klassVtable::is_miranda(Method* m, Array* class_methods, && mo->method_holder() != NULL && mo->method_holder()->super() != NULL) { - mo = mo->method_holder()->super()->uncached_lookup_method(name, signature, Klass::normal); + mo = mo->method_holder()->super()->uncached_lookup_method(name, signature, Klass::find_overpass); } if (mo == NULL || mo->access_flags().is_private() ) { // super class hierarchy does not implement it or protection is different @@ -770,7 +770,7 @@ void klassVtable::add_new_mirandas_to_lists( if (is_miranda(im, class_methods, default_methods, super)) { // is it a miranda at all? InstanceKlass *sk = InstanceKlass::cast(super); // check if it is a duplicate of a super's miranda - if (sk->lookup_method_in_all_interfaces(im->name(), im->signature(), Klass::normal) == NULL) { + if (sk->lookup_method_in_all_interfaces(im->name(), im->signature(), Klass::find_defaults) == NULL) { new_mirandas->append(im); } if (all_mirandas != NULL) { diff --git a/hotspot/src/share/vm/prims/jvm.cpp b/hotspot/src/share/vm/prims/jvm.cpp index a94242a64a1..4f857d6e0af 100644 --- a/hotspot/src/share/vm/prims/jvm.cpp +++ b/hotspot/src/share/vm/prims/jvm.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -1167,7 +1167,7 @@ JVM_ENTRY(jobject, JVM_DoPrivileged(JNIEnv *env, jclass cls, jobject action, job Method* m_oop = object->klass()->uncached_lookup_method( vmSymbols::run_method_name(), vmSymbols::void_object_signature(), - Klass::normal); + Klass::find_overpass); methodHandle m (THREAD, m_oop); if (m.is_null() || !m->is_method() || !m()->is_public() || m()->is_static()) { THROW_MSG_0(vmSymbols::java_lang_InternalError(), "No run method"); diff --git a/hotspot/src/share/vm/prims/nativeLookup.cpp b/hotspot/src/share/vm/prims/nativeLookup.cpp index 35d4704081b..89b57e272d0 100644 --- a/hotspot/src/share/vm/prims/nativeLookup.cpp +++ b/hotspot/src/share/vm/prims/nativeLookup.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -393,7 +393,7 @@ address NativeLookup::base_library_lookup(const char* class_name, const char* me // Find method and invoke standard lookup methodHandle method (THREAD, - klass->uncached_lookup_method(m_name, s_name, Klass::normal)); + klass->uncached_lookup_method(m_name, s_name, Klass::find_overpass)); address result = lookup(method, in_base_library, CATCH); assert(in_base_library, "must be in basic library"); guarantee(result != NULL, "must be non NULL"); From 2c2593b7574bb0521567b194f6ce02153228f348 Mon Sep 17 00:00:00 2001 From: Christian Tornqvist Date: Fri, 23 Jan 2015 11:44:21 -0800 Subject: [PATCH 02/58] 8069207: [TESTBUG] Exception thrown for java.lang.NoSuchMethodError: sun.misc.Unsafe.monitorExit Reviewed-by: gtriantafill, dholmes --- .../test/runtime/Unsafe/AllocateInstance.java | 63 ++++++++++++ .../test/runtime/Unsafe/AllocateMemory.java | 66 +++++++++++++ hotspot/test/runtime/Unsafe/CopyMemory.java | 56 +++++++++++ hotspot/test/runtime/Unsafe/DefineClass.java | 99 +++++++++++++++++++ hotspot/test/runtime/Unsafe/FieldOffset.java | 65 ++++++++++++ hotspot/test/runtime/Unsafe/GetField.java | 46 +++++++++ .../test/runtime/Unsafe/GetPutAddress.java | 52 ++++++++++ .../test/runtime/Unsafe/GetPutBoolean.java | 59 +++++++++++ hotspot/test/runtime/Unsafe/GetPutByte.java | 64 ++++++++++++ hotspot/test/runtime/Unsafe/GetPutChar.java | 64 ++++++++++++ hotspot/test/runtime/Unsafe/GetPutDouble.java | 64 ++++++++++++ hotspot/test/runtime/Unsafe/GetPutFloat.java | 64 ++++++++++++ hotspot/test/runtime/Unsafe/GetPutInt.java | 63 ++++++++++++ hotspot/test/runtime/Unsafe/GetPutLong.java | 64 ++++++++++++ hotspot/test/runtime/Unsafe/GetPutObject.java | 61 ++++++++++++ hotspot/test/runtime/Unsafe/GetPutShort.java | 64 ++++++++++++ hotspot/test/runtime/Unsafe/GetUnsafe.java | 44 +++++++++ hotspot/test/runtime/Unsafe/PageSize.java | 48 +++++++++ hotspot/test/runtime/Unsafe/SetMemory.java | 44 +++++++++ .../test/runtime/Unsafe/ThrowException.java | 49 +++++++++ 20 files changed, 1199 insertions(+) create mode 100644 hotspot/test/runtime/Unsafe/AllocateInstance.java create mode 100644 hotspot/test/runtime/Unsafe/AllocateMemory.java create mode 100644 hotspot/test/runtime/Unsafe/CopyMemory.java create mode 100644 hotspot/test/runtime/Unsafe/DefineClass.java create mode 100644 hotspot/test/runtime/Unsafe/FieldOffset.java create mode 100644 hotspot/test/runtime/Unsafe/GetField.java create mode 100644 hotspot/test/runtime/Unsafe/GetPutAddress.java create mode 100644 hotspot/test/runtime/Unsafe/GetPutBoolean.java create mode 100644 hotspot/test/runtime/Unsafe/GetPutByte.java create mode 100644 hotspot/test/runtime/Unsafe/GetPutChar.java create mode 100644 hotspot/test/runtime/Unsafe/GetPutDouble.java create mode 100644 hotspot/test/runtime/Unsafe/GetPutFloat.java create mode 100644 hotspot/test/runtime/Unsafe/GetPutInt.java create mode 100644 hotspot/test/runtime/Unsafe/GetPutLong.java create mode 100644 hotspot/test/runtime/Unsafe/GetPutObject.java create mode 100644 hotspot/test/runtime/Unsafe/GetPutShort.java create mode 100644 hotspot/test/runtime/Unsafe/GetUnsafe.java create mode 100644 hotspot/test/runtime/Unsafe/PageSize.java create mode 100644 hotspot/test/runtime/Unsafe/SetMemory.java create mode 100644 hotspot/test/runtime/Unsafe/ThrowException.java diff --git a/hotspot/test/runtime/Unsafe/AllocateInstance.java b/hotspot/test/runtime/Unsafe/AllocateInstance.java new file mode 100644 index 00000000000..1339572e1d0 --- /dev/null +++ b/hotspot/test/runtime/Unsafe/AllocateInstance.java @@ -0,0 +1,63 @@ +/* + * Copyright (c) 2015, 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 + * @summary Verifies the behaviour of Unsafe.allocateInstance + * @library /testlibrary + * @run main AllocateInstance + */ + +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import static com.oracle.java.testlibrary.Asserts.*; + +public class AllocateInstance { + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + + // allocateInstance() should not result in a call to the constructor + TestClass tc = (TestClass)unsafe.allocateInstance(TestClass.class); + assertFalse(tc.calledConstructor); + + // allocateInstance() on an abstract class should result in an InstantiationException + try { + AbstractClass ac = (AbstractClass)unsafe.allocateInstance(AbstractClass.class); + throw new RuntimeException("Did not get expected InstantiationException"); + } catch (InstantiationException e) { + // Expected + } + } + + class TestClass { + public boolean calledConstructor = false; + + public TestClass() { + calledConstructor = true; + } + } + + abstract class AbstractClass { + public AbstractClass() {} + } +} diff --git a/hotspot/test/runtime/Unsafe/AllocateMemory.java b/hotspot/test/runtime/Unsafe/AllocateMemory.java new file mode 100644 index 00000000000..0c2b28f3911 --- /dev/null +++ b/hotspot/test/runtime/Unsafe/AllocateMemory.java @@ -0,0 +1,66 @@ +/* + * Copyright (c) 2015, 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 + * @summary Verifies behaviour of Unsafe.allocateMemory + * @library /testlibrary + * @run main/othervm -XX:+UnlockDiagnosticVMOptions -XX:MallocMaxTestWords=20m AllocateMemory + */ + +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import static com.oracle.java.testlibrary.Asserts.*; + +public class AllocateMemory { + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + + // Allocate a byte, write to the location and read back the value + long address = unsafe.allocateMemory(1); + assertNotEquals(address, 0L); + + unsafe.putByte(address, Byte.MAX_VALUE); + assertEquals(Byte.MAX_VALUE, unsafe.getByte(address)); + unsafe.freeMemory(address); + + // Call to allocateMemory() with a negative value should result in an IllegalArgumentException + try { + address = unsafe.allocateMemory(-1); + throw new RuntimeException("Did not get expected IllegalArgumentException"); + } catch (IllegalArgumentException e) { + // Expected + assertNotEquals(address, 0L); + } + + // allocateMemory() should throw an OutOfMemoryError when the underlying malloc fails, + // we test this by limiting the malloc using -XX:MallocMaxTestWords + try { + address = unsafe.allocateMemory(20 * 1024 * 1024 * 8); + } catch (OutOfMemoryError e) { + // Expected + return; + } + throw new RuntimeException("Did not get expected OutOfMemoryError"); + } +} diff --git a/hotspot/test/runtime/Unsafe/CopyMemory.java b/hotspot/test/runtime/Unsafe/CopyMemory.java new file mode 100644 index 00000000000..c5b1ac49eb2 --- /dev/null +++ b/hotspot/test/runtime/Unsafe/CopyMemory.java @@ -0,0 +1,56 @@ +/* + * Copyright (c) 2015, 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 + * @summary Verifies behaviour of Unsafe.copyMemory + * @library /testlibrary + * @run main CopyMemory + */ + +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import static com.oracle.java.testlibrary.Asserts.*; + +public class CopyMemory { + final static int LENGTH = 8; + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + long src = unsafe.allocateMemory(LENGTH); + long dst = unsafe.allocateMemory(LENGTH); + assertNotEquals(src, 0L); + assertNotEquals(dst, 0L); + + // call copyMemory() with different lengths and verify the contents of + // the destination array + for (int i = 0; i < LENGTH; i++) { + unsafe.putByte(src + i, (byte)i); + unsafe.copyMemory(src, dst, i); + for (int j = 0; j < i; j++) { + assertEquals(unsafe.getByte(src + j), unsafe.getByte(src + j)); + } + } + unsafe.freeMemory(src); + unsafe.freeMemory(dst); + } +} diff --git a/hotspot/test/runtime/Unsafe/DefineClass.java b/hotspot/test/runtime/Unsafe/DefineClass.java new file mode 100644 index 00000000000..a22bc57a8b1 --- /dev/null +++ b/hotspot/test/runtime/Unsafe/DefineClass.java @@ -0,0 +1,99 @@ +/* + * Copyright (c) 2015, 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 + * @summary Verifies the behaviour of Unsafe.defineClass + * @library /testlibrary + * @run main DefineClass + */ + +import java.security.ProtectionDomain; +import java.io.InputStream; +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import static com.oracle.java.testlibrary.Asserts.*; + +public class DefineClass { + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + TestClassLoader classloader = new TestClassLoader(); + ProtectionDomain pd = new ProtectionDomain(null, null); + + byte klassbuf[] = InMemoryJavaCompiler.compile("TestClass", "class TestClass { }"); + + // Invalid class data + try { + unsafe.defineClass(null, klassbuf, 4, klassbuf.length - 4, classloader, pd); + throw new RuntimeException("defineClass did not throw expected ClassFormatError"); + } catch (ClassFormatError e) { + // Expected + } + + // Negative offset + try { + unsafe.defineClass(null, klassbuf, -1, klassbuf.length, classloader, pd); + throw new RuntimeException("defineClass did not throw expected IndexOutOfBoundsException"); + } catch (IndexOutOfBoundsException e) { + // Expected + } + + // Negative length + try { + unsafe.defineClass(null, klassbuf, 0, -1, classloader, pd); + throw new RuntimeException("defineClass did not throw expected IndexOutOfBoundsException"); + } catch (IndexOutOfBoundsException e) { + // Expected + } + + // Offset greater than klassbuf.length + try { + unsafe.defineClass(null, klassbuf, klassbuf.length + 1, klassbuf.length, classloader, pd); + throw new RuntimeException("defineClass did not throw expected IndexOutOfBoundsException"); + } catch (IndexOutOfBoundsException e) { + // Expected + } + + // Length greater than klassbuf.length + try { + unsafe.defineClass(null, klassbuf, 0, klassbuf.length + 1, classloader, pd); + throw new RuntimeException("defineClass did not throw expected IndexOutOfBoundsException"); + } catch (IndexOutOfBoundsException e) { + // Expected + } + + Class klass = unsafe.defineClass(null, klassbuf, 0, klassbuf.length, classloader, pd); + assertEquals(klass.getClassLoader(), classloader); + assertEquals(klass.getProtectionDomain(), pd); + } + + private static class TestClassLoader extends ClassLoader { + public TestClassLoader(ClassLoader parent) { + super(parent); + } + + public TestClassLoader() { + super(); + } + } +} diff --git a/hotspot/test/runtime/Unsafe/FieldOffset.java b/hotspot/test/runtime/Unsafe/FieldOffset.java new file mode 100644 index 00000000000..7eb609f8da2 --- /dev/null +++ b/hotspot/test/runtime/Unsafe/FieldOffset.java @@ -0,0 +1,65 @@ +/* + * Copyright (c) 2015, 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 + * @summary Verifies the behaviour of Unsafe.fieldOffset + * @library /testlibrary + * @run main FieldOffset + */ + +import java.lang.reflect.Field; +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import java.lang.reflect.*; +import static com.oracle.java.testlibrary.Asserts.*; + +public class FieldOffset { + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + Field fields[] = Test.class.getDeclaredFields(); + + for (int i = 0; i < fields.length; i++) { + int offset = unsafe.fieldOffset(fields[i]); + // Ensure we got a valid offset value back + assertNotEquals(offset, unsafe.INVALID_FIELD_OFFSET); + + // Make sure the field offset is unique + for (int j = 0; j < i; j++) { + assertNotEquals(offset, unsafe.fieldOffset(fields[j])); + } + } + } + + class Test { + boolean booleanField; + byte byteField; + char charField; + double doubleField; + float floatField; + int intField; + long longField; + Object objectField; + short shortField; + } +} diff --git a/hotspot/test/runtime/Unsafe/GetField.java b/hotspot/test/runtime/Unsafe/GetField.java new file mode 100644 index 00000000000..d5e58c9d8c6 --- /dev/null +++ b/hotspot/test/runtime/Unsafe/GetField.java @@ -0,0 +1,46 @@ +/* + * Copyright (c) 2015, 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 + * @summary Verifies behaviour of Unsafe.getField + * @library /testlibrary + * @run main GetField + */ + +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import java.lang.reflect.*; +import static com.oracle.java.testlibrary.Asserts.*; + +public class GetField { + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + // Unsafe.INVALID_FIELD_OFFSET is a static final int field, + // make sure getField returns the correct field + Field field = Unsafe.class.getField("INVALID_FIELD_OFFSET"); + assertNotEquals(field.getModifiers() & Modifier.FINAL, 0); + assertNotEquals(field.getModifiers() & Modifier.STATIC, 0); + assertEquals(field.getType(), int.class); + } +} diff --git a/hotspot/test/runtime/Unsafe/GetPutAddress.java b/hotspot/test/runtime/Unsafe/GetPutAddress.java new file mode 100644 index 00000000000..9bf105b7513 --- /dev/null +++ b/hotspot/test/runtime/Unsafe/GetPutAddress.java @@ -0,0 +1,52 @@ +/* + * Copyright (c) 2015, 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 + * Verify behaviour of Unsafe.get/putAddress and Unsafe.addressSize + * @library /testlibrary + * @run main GetPutAddress + */ + +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import static com.oracle.java.testlibrary.Asserts.*; + +public class GetPutAddress { + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + int addressSize = unsafe.addressSize(); + // Ensure the size returned from Unsafe.addressSize is correct + assertEquals(unsafe.addressSize(), Platform.is32bit() ? 4 : 8); + + // Write the address, read it back and make sure it's the same value + long address = unsafe.allocateMemory(addressSize); + unsafe.putAddress(address, address); + long readAddress = unsafe.getAddress(address); + if (addressSize == 4) { + readAddress &= 0x00000000FFFFFFFFL; + } + assertEquals(address, readAddress); + unsafe.freeMemory(address); + } +} diff --git a/hotspot/test/runtime/Unsafe/GetPutBoolean.java b/hotspot/test/runtime/Unsafe/GetPutBoolean.java new file mode 100644 index 00000000000..d7b64f83a31 --- /dev/null +++ b/hotspot/test/runtime/Unsafe/GetPutBoolean.java @@ -0,0 +1,59 @@ +/* + * Copyright (c) 2015, 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 + * @summary Verify behaviour of Unsafe.get/putBoolean + * @library /testlibrary + * @run main GetPutBoolean + */ + +import java.lang.reflect.Field; +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import static com.oracle.java.testlibrary.Asserts.*; + +public class GetPutBoolean { + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + Test t = new Test(); + Field field = Test.class.getField("b1"); + + int offset = unsafe.fieldOffset(field); + assertEquals(false, unsafe.getBoolean(t, offset)); + unsafe.putBoolean(t, offset, true); + assertEquals(true, unsafe.getBoolean(t, offset)); + + boolean arrayBoolean[] = { true, false, false, true }; + int scale = unsafe.arrayIndexScale(arrayBoolean.getClass()); + offset = unsafe.arrayBaseOffset(arrayBoolean.getClass()); + for (int i = 0; i < arrayBoolean.length; i++) { + assertEquals(unsafe.getBoolean(arrayBoolean, offset), arrayBoolean[i]); + offset += scale; + } + } + + static class Test { + public boolean b1 = false; + } +} diff --git a/hotspot/test/runtime/Unsafe/GetPutByte.java b/hotspot/test/runtime/Unsafe/GetPutByte.java new file mode 100644 index 00000000000..016a2a94049 --- /dev/null +++ b/hotspot/test/runtime/Unsafe/GetPutByte.java @@ -0,0 +1,64 @@ +/* + * Copyright (c) 2015, 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 + * @summary Verify behaviour of Unsafe.get/putByte + * @library /testlibrary + * @run main GetPutByte + */ + +import java.lang.reflect.Field; +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import static com.oracle.java.testlibrary.Asserts.*; + +public class GetPutByte { + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + Test t = new Test(); + Field field = Test.class.getField("b"); + + int offset = unsafe.fieldOffset(field); + assertEquals((byte)0, unsafe.getByte(t, offset)); + unsafe.putByte(t, offset, (byte)1); + assertEquals((byte)1, unsafe.getByte(t, offset)); + + long address = unsafe.allocateMemory(8); + unsafe.putByte(address, (byte)2); + assertEquals((byte)2, unsafe.getByte(address)); + unsafe.freeMemory(address); + + byte arrayByte[] = { -1, 0, 1, 2 }; + int scale = unsafe.arrayIndexScale(arrayByte.getClass()); + offset = unsafe.arrayBaseOffset(arrayByte.getClass()); + for (int i = 0; i < arrayByte.length; i++) { + assertEquals(unsafe.getByte(arrayByte, offset), arrayByte[i]); + offset += scale; + } + } + + static class Test { + public byte b = 0; + } +} diff --git a/hotspot/test/runtime/Unsafe/GetPutChar.java b/hotspot/test/runtime/Unsafe/GetPutChar.java new file mode 100644 index 00000000000..83dd737f218 --- /dev/null +++ b/hotspot/test/runtime/Unsafe/GetPutChar.java @@ -0,0 +1,64 @@ +/* + * Copyright (c) 2015, 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 + * @summary Verify behaviour of Unsafe.get/putChar + * @library /testlibrary + * @run main GetPutChar + */ + +import java.lang.reflect.Field; +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import static com.oracle.java.testlibrary.Asserts.*; + +public class GetPutChar { + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + Test t = new Test(); + Field field = Test.class.getField("c"); + + int offset = unsafe.fieldOffset(field); + assertEquals('\u0000', unsafe.getChar(t, offset)); + unsafe.putChar(t, offset, '\u0001'); + assertEquals('\u0001', unsafe.getChar(t, offset)); + + long address = unsafe.allocateMemory(8); + unsafe.putChar(address, '\u0002'); + assertEquals('\u0002', unsafe.getChar(address)); + unsafe.freeMemory(address); + + char arrayChar[] = { '\uabcd', '\u00ff', '\uff00', }; + int scale = unsafe.arrayIndexScale(arrayChar.getClass()); + offset = unsafe.arrayBaseOffset(arrayChar.getClass()); + for (int i = 0; i < arrayChar.length; i++) { + assertEquals(unsafe.getChar(arrayChar, offset), arrayChar[i]); + offset += scale; + } + } + + static class Test { + public char c = '\u0000'; + } +} diff --git a/hotspot/test/runtime/Unsafe/GetPutDouble.java b/hotspot/test/runtime/Unsafe/GetPutDouble.java new file mode 100644 index 00000000000..dca8d3038c9 --- /dev/null +++ b/hotspot/test/runtime/Unsafe/GetPutDouble.java @@ -0,0 +1,64 @@ +/* + * Copyright (c) 2015, 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 + * @summary Verify behaviour of Unsafe.get/putDouble + * @library /testlibrary + * @run main GetPutDouble + */ + +import java.lang.reflect.Field; +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import static com.oracle.java.testlibrary.Asserts.*; + +public class GetPutDouble { + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + Test t = new Test(); + Field field = Test.class.getField("d"); + + int offset = unsafe.fieldOffset(field); + assertEquals(-1.0, unsafe.getDouble(t, offset)); + unsafe.putDouble(t, offset, 0.0); + assertEquals(0.0, unsafe.getDouble(t, offset)); + + long address = unsafe.allocateMemory(8); + unsafe.putDouble(address, 1.0); + assertEquals(1.0, unsafe.getDouble(address)); + unsafe.freeMemory(address); + + double arrayDouble[] = { -1.0, 0.0, 1.0, 2.0 }; + int scale = unsafe.arrayIndexScale(arrayDouble.getClass()); + offset = unsafe.arrayBaseOffset(arrayDouble.getClass()); + for (int i = 0; i < arrayDouble.length; i++) { + assertEquals(unsafe.getDouble(arrayDouble, offset), arrayDouble[i]); + offset += scale; + } + } + + static class Test { + public double d = -1.0; + } +} diff --git a/hotspot/test/runtime/Unsafe/GetPutFloat.java b/hotspot/test/runtime/Unsafe/GetPutFloat.java new file mode 100644 index 00000000000..64125671721 --- /dev/null +++ b/hotspot/test/runtime/Unsafe/GetPutFloat.java @@ -0,0 +1,64 @@ +/* + * Copyright (c) 2015, 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 + * @summary Verify behaviour of Unsafe.get/putFloat + * @library /testlibrary + * @run main GetPutFloat + */ + +import java.lang.reflect.Field; +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import static com.oracle.java.testlibrary.Asserts.*; + +public class GetPutFloat { + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + Test t = new Test(); + Field field = Test.class.getField("f"); + + int offset = unsafe.fieldOffset(field); + assertEquals(-1.0f, unsafe.getFloat(t, offset)); + unsafe.putFloat(t, offset, 0.0f); + assertEquals(0.0f, unsafe.getFloat(t, offset)); + + long address = unsafe.allocateMemory(8); + unsafe.putFloat(address, 1.0f); + assertEquals(1.0f, unsafe.getFloat(address)); + unsafe.freeMemory(address); + + float arrayFloat[] = { -1.0f, 0.0f, 1.0f, 2.0f }; + int scale = unsafe.arrayIndexScale(arrayFloat.getClass()); + offset = unsafe.arrayBaseOffset(arrayFloat.getClass()); + for (int i = 0; i < arrayFloat.length; i++) { + assertEquals(unsafe.getFloat(arrayFloat, offset), arrayFloat[i]); + offset += scale; + } + } + + static class Test { + public float f = -1.0f; + } +} diff --git a/hotspot/test/runtime/Unsafe/GetPutInt.java b/hotspot/test/runtime/Unsafe/GetPutInt.java new file mode 100644 index 00000000000..9239c073bff --- /dev/null +++ b/hotspot/test/runtime/Unsafe/GetPutInt.java @@ -0,0 +1,63 @@ +/* + * Copyright (c) 2015, 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 + * @library /testlibrary + * @run main GetPutInt + */ + +import java.lang.reflect.Field; +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import static com.oracle.java.testlibrary.Asserts.*; + +public class GetPutInt { + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + Test t = new Test(); + Field field = Test.class.getField("i"); + + int offset = unsafe.fieldOffset(field); + assertEquals(-1, unsafe.getInt(t, offset)); + unsafe.putInt(t, offset, 0); + assertEquals(0, unsafe.getInt(t, offset)); + + long address = unsafe.allocateMemory(8); + unsafe.putInt(address, 1); + assertEquals(1, unsafe.getInt(address)); + unsafe.freeMemory(address); + + int arrayInt[] = { -1, 0, 1, 2 }; + int scale = unsafe.arrayIndexScale(arrayInt.getClass()); + offset = unsafe.arrayBaseOffset(arrayInt.getClass()); + for (int i = 0; i < arrayInt.length; i++) { + assertEquals(unsafe.getInt(arrayInt, offset), arrayInt[i]); + offset += scale; + } + } + + static class Test { + public int i = -1; + } +} diff --git a/hotspot/test/runtime/Unsafe/GetPutLong.java b/hotspot/test/runtime/Unsafe/GetPutLong.java new file mode 100644 index 00000000000..ae8fe5f3f4d --- /dev/null +++ b/hotspot/test/runtime/Unsafe/GetPutLong.java @@ -0,0 +1,64 @@ +/* + * Copyright (c) 2015, 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 + * @summary Verify behaviour of Unsafe.get/putLong + * @library /testlibrary + * @run main GetPutLong + */ + +import java.lang.reflect.Field; +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import static com.oracle.java.testlibrary.Asserts.*; + +public class GetPutLong { + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + Test t = new Test(); + Field field = Test.class.getField("l"); + + int offset = unsafe.fieldOffset(field); + assertEquals(-1L, unsafe.getLong(t, offset)); + unsafe.putLong(t, offset, 0L); + assertEquals(0L, unsafe.getLong(t, offset)); + + long address = unsafe.allocateMemory(8); + unsafe.putLong(address, 1L); + assertEquals(1L, unsafe.getLong(address)); + unsafe.freeMemory(address); + + long arrayLong[] = { -1, 0, 1, 2 }; + int scale = unsafe.arrayIndexScale(arrayLong.getClass()); + offset = unsafe.arrayBaseOffset(arrayLong.getClass()); + for (int i = 0; i < arrayLong.length; i++) { + assertEquals(unsafe.getLong(arrayLong, offset), arrayLong[i]); + offset += scale; + } + } + + static class Test { + public long l = -1L; + } +} diff --git a/hotspot/test/runtime/Unsafe/GetPutObject.java b/hotspot/test/runtime/Unsafe/GetPutObject.java new file mode 100644 index 00000000000..03813acf087 --- /dev/null +++ b/hotspot/test/runtime/Unsafe/GetPutObject.java @@ -0,0 +1,61 @@ +/* + * Copyright (c) 2015, 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 + * @summary Verify behaviour of Unsafe.get/putObject + * @library /testlibrary + * @run main GetPutObject + */ + +import java.lang.reflect.Field; +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import static com.oracle.java.testlibrary.Asserts.*; + +public class GetPutObject { + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + Test t = new Test(); + Object o = new Object(); + Field field = Test.class.getField("o"); + + int offset = unsafe.fieldOffset(field); + assertEquals(t.o, unsafe.getObject(t, offset)); + + unsafe.putObject(t, offset, o); + assertEquals(o, unsafe.getObject(t, offset)); + + Object arrayObject[] = { unsafe, null, new Object() }; + int scale = unsafe.arrayIndexScale(arrayObject.getClass()); + offset = unsafe.arrayBaseOffset(arrayObject.getClass()); + for (int i = 0; i < arrayObject.length; i++) { + assertEquals(unsafe.getObject(arrayObject, offset), arrayObject[i]); + offset += scale; + } + } + + static class Test { + public Object o = new Object(); + } +} diff --git a/hotspot/test/runtime/Unsafe/GetPutShort.java b/hotspot/test/runtime/Unsafe/GetPutShort.java new file mode 100644 index 00000000000..263cde28f3c --- /dev/null +++ b/hotspot/test/runtime/Unsafe/GetPutShort.java @@ -0,0 +1,64 @@ +/* + * Copyright (c) 2015, 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 + * @summary Verify behaviour of Unsafe.get/putShort + * @library /testlibrary + * @run main GetPutShort + */ + +import java.lang.reflect.Field; +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import static com.oracle.java.testlibrary.Asserts.*; + +public class GetPutShort { + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + Test t = new Test(); + Field field = Test.class.getField("s"); + + int offset = unsafe.fieldOffset(field); + assertEquals((short)-1, unsafe.getShort(t, offset)); + unsafe.putShort(t, offset, (short)0); + assertEquals((short)0, unsafe.getShort(t, offset)); + + long address = unsafe.allocateMemory(8); + unsafe.putShort(address, (short)1); + assertEquals((short)1, unsafe.getShort(address)); + unsafe.freeMemory(address); + + short arrayShort[] = { -1, 0, 1, 2 }; + int scale = unsafe.arrayIndexScale(arrayShort.getClass()); + offset = unsafe.arrayBaseOffset(arrayShort.getClass()); + for (int i = 0; i < arrayShort.length; i++) { + assertEquals(unsafe.getShort(arrayShort, offset), arrayShort[i]); + offset += scale; + } + } + + static class Test { + public short s = -1; + } +} diff --git a/hotspot/test/runtime/Unsafe/GetUnsafe.java b/hotspot/test/runtime/Unsafe/GetUnsafe.java new file mode 100644 index 00000000000..14c6880e6e2 --- /dev/null +++ b/hotspot/test/runtime/Unsafe/GetUnsafe.java @@ -0,0 +1,44 @@ +/* + * Copyright (c) 2015, 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 + * @summary Verifies that getUnsafe() actually throws SecurityException when unsafeAccess is prohibited. + * @library /testlibrary + * @run main GetUnsafe + */ + +import sun.misc.Unsafe; +import static com.oracle.java.testlibrary.Asserts.*; + +public class GetUnsafe { + public static void main(String args[]) throws Exception { + try { + Unsafe unsafe = Unsafe.getUnsafe(); + } catch (SecurityException e) { + // Expected + return; + } + throw new RuntimeException("Did not get expected SecurityException"); + } +} diff --git a/hotspot/test/runtime/Unsafe/PageSize.java b/hotspot/test/runtime/Unsafe/PageSize.java new file mode 100644 index 00000000000..786277c8974 --- /dev/null +++ b/hotspot/test/runtime/Unsafe/PageSize.java @@ -0,0 +1,48 @@ +/* + * Copyright (c) 2015, 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 + * @summary Make sure pageSize() returns a value that is a power of two + * @library /testlibrary + * @run main PageSize + */ + +import java.lang.reflect.Field; +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import static com.oracle.java.testlibrary.Asserts.*; + +public class PageSize { + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + int pageSize = unsafe.pageSize(); + + for (int n = 1; n != 0; n <<= 1) { + if (pageSize == n) { + return; + } + } + throw new RuntimeException("Expected pagesize to be a power of two, actual pagesize:" + pageSize); + } +} diff --git a/hotspot/test/runtime/Unsafe/SetMemory.java b/hotspot/test/runtime/Unsafe/SetMemory.java new file mode 100644 index 00000000000..ea7c03f84e3 --- /dev/null +++ b/hotspot/test/runtime/Unsafe/SetMemory.java @@ -0,0 +1,44 @@ +/* + * Copyright (c) 2015, 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 + * @summary Verifies that setMemory works correctly + * @library /testlibrary + * @run main SetMemory + */ + +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import static com.oracle.java.testlibrary.Asserts.*; + +public class SetMemory { + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + long address = unsafe.allocateMemory(1); + assertNotEquals(address, 0L); + unsafe.setMemory(address, 1, (byte)17); + assertEquals((byte)17, unsafe.getByte(address)); + unsafe.freeMemory(address); + } +} diff --git a/hotspot/test/runtime/Unsafe/ThrowException.java b/hotspot/test/runtime/Unsafe/ThrowException.java new file mode 100644 index 00000000000..d0347b542ce --- /dev/null +++ b/hotspot/test/runtime/Unsafe/ThrowException.java @@ -0,0 +1,49 @@ +/* + * Copyright (c) 2015, 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 + * @summary Verify that throwException() can throw an exception + * @library /testlibrary + * @run main ThrowException + */ + +import com.oracle.java.testlibrary.*; +import sun.misc.Unsafe; +import static com.oracle.java.testlibrary.Asserts.*; + +public class ThrowException { + public static void main(String args[]) throws Exception { + Unsafe unsafe = Utils.getUnsafe(); + try { + unsafe.throwException(new TestException()); + } catch (Throwable t) { + if (t instanceof TestException) { + return; + } + throw t; + } + throw new RuntimeException("Did not throw expected TestException"); + } + static class TestException extends Exception {} +} From 2ffd52f3068236f80a76bc6d851bad92b95313b9 Mon Sep 17 00:00:00 2001 From: Goetz Lindenmaier Date: Mon, 12 Jan 2015 09:16:06 +0100 Subject: [PATCH 03/58] 8068778: [TESTBUG] CompressedClassSpaceSizeInJmapHeap.java fails if SA not available Reviewed-by: coleenp, sla --- .../test/gc/metaspace/CompressedClassSpaceSizeInJmapHeap.java | 4 ++++ 1 file changed, 4 insertions(+) diff --git a/hotspot/test/gc/metaspace/CompressedClassSpaceSizeInJmapHeap.java b/hotspot/test/gc/metaspace/CompressedClassSpaceSizeInJmapHeap.java index 489ae411669..8fb4ee559ee 100644 --- a/hotspot/test/gc/metaspace/CompressedClassSpaceSizeInJmapHeap.java +++ b/hotspot/test/gc/metaspace/CompressedClassSpaceSizeInJmapHeap.java @@ -41,6 +41,10 @@ public class CompressedClassSpaceSizeInJmapHeap { // Compressed Class Space is only available on 64-bit JVMs return; } + if (!Platform.shouldSAAttach()) { + System.out.println("SA attach not expected to work - test skipped."); + return; + } String pid = Integer.toString(ProcessTools.getProcessId()); From 1249ca93f99237bd55e340e58410004bb34ce1a6 Mon Sep 17 00:00:00 2001 From: Filipp Zhinkin Date: Tue, 20 Jan 2015 11:26:20 +0300 Subject: [PATCH 04/58] 8069126: compiler/rtm/locking/TestRTMTotalCountIncrRate.java nightly failure Reviewed-by: kvn --- .../compiler/rtm/locking/TestRTMTotalCountIncrRate.java | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/hotspot/test/compiler/rtm/locking/TestRTMTotalCountIncrRate.java b/hotspot/test/compiler/rtm/locking/TestRTMTotalCountIncrRate.java index d27a87d8c2e..56a66d0804f 100644 --- a/hotspot/test/compiler/rtm/locking/TestRTMTotalCountIncrRate.java +++ b/hotspot/test/compiler/rtm/locking/TestRTMTotalCountIncrRate.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2014, 2015, 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 @@ -119,13 +119,13 @@ public class TestRTMTotalCountIncrRate extends CommandLineOptionTest { return new String[] { getMethodWithLockName() }; } - public void lock(booleab forceAbort) { + public void lock(boolean forceAbort) { synchronized(monitor) { if (forceAbort) { // We're calling native method in order to force // abort. It's done by explicit xabort call emitted // in SharedRuntime::generate_native_wrapper. - // If an actuall JNI call will be replaced by + // If an actual JNI call will be replaced by // intrinsic - we'll be in trouble, since xabort // will be no longer called and test may fail. UNSAFE.addressSize(); From 8e93b53e3e5994fb42f938a0f485eaec087c37c5 Mon Sep 17 00:00:00 2001 From: Katja Kantserova Date: Wed, 21 Jan 2015 15:06:31 +0100 Subject: [PATCH 05/58] 8069296: java/lang/management/MemoryMXBean/LowMemoryTest.java should be quarantined Reviewed-by: sla --- jdk/test/ProblemList.txt | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/jdk/test/ProblemList.txt b/jdk/test/ProblemList.txt index c6c35f2e4d6..4954f772f8a 100644 --- a/jdk/test/ProblemList.txt +++ b/jdk/test/ProblemList.txt @@ -123,7 +123,6 @@ # 8029891 java/lang/ClassLoader/deadlock/GetResource.java generic-all - ############################################################################ # jdk_instrument @@ -142,6 +141,9 @@ java/lang/instrument/RetransformBigClass.sh generic-all # 8058492 java/lang/management/ThreadMXBean/FindDeadlocks.java generic-all +# 8069286 +java/lang/management/MemoryMXBean/LowMemoryTest.java generic-all + ############################################################################ # jdk_jmx From 21e2e5ffc05df90741080218ad6b908a4cc6921d Mon Sep 17 00:00:00 2001 From: Tobias Hartmann Date: Thu, 22 Jan 2015 09:07:41 +0100 Subject: [PATCH 06/58] 8069580: String intrinsic related cleanups Small cleanup of string intrinsic related code. Reviewed-by: kvn, roland --- hotspot/src/cpu/sparc/vm/sparc.ad | 4 ++-- hotspot/src/cpu/x86/vm/macroAssembler_x86.cpp | 9 ++++----- hotspot/src/share/vm/opto/library_call.cpp | 8 +------- 3 files changed, 7 insertions(+), 14 deletions(-) diff --git a/hotspot/src/cpu/sparc/vm/sparc.ad b/hotspot/src/cpu/sparc/vm/sparc.ad index dfc67d5df66..739cc9612bd 100644 --- a/hotspot/src/cpu/sparc/vm/sparc.ad +++ b/hotspot/src/cpu/sparc/vm/sparc.ad @@ -1,5 +1,5 @@ // -// Copyright (c) 1998, 2013, Oracle and/or its affiliates. All rights reserved. +// Copyright (c) 1998, 2015, 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 @@ -2996,7 +2996,7 @@ enc_class Fast_Unlock(iRegP oop, iRegP box, o7RegP scratch, iRegP scratch2) %{ %} enc_class enc_String_Equals(o0RegP str1, o1RegP str2, g3RegI cnt, notemp_iRegI result) %{ - Label Lword_loop, Lpost_word, Lchar, Lchar_loop, Ldone; + Label Lchar, Lchar_loop, Ldone; MacroAssembler _masm(&cbuf); Register str1_reg = reg_to_register_object($str1$$reg); diff --git a/hotspot/src/cpu/x86/vm/macroAssembler_x86.cpp b/hotspot/src/cpu/x86/vm/macroAssembler_x86.cpp index 24587840f98..2cb6bd820d0 100644 --- a/hotspot/src/cpu/x86/vm/macroAssembler_x86.cpp +++ b/hotspot/src/cpu/x86/vm/macroAssembler_x86.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -6194,7 +6194,7 @@ void MacroAssembler::string_indexofC8(Register str1, Register str2, ShortBranchVerifier sbv(this); assert(UseSSE42Intrinsics, "SSE4.2 is required"); - // This method uses pcmpestri inxtruction with bound registers + // This method uses pcmpestri instruction with bound registers // inputs: // xmm - substring // rax - substring length (elements count) @@ -6355,7 +6355,7 @@ void MacroAssembler::string_indexof(Register str1, Register str2, // assert(int_cnt2 == -1 || (0 < int_cnt2 && int_cnt2 < 8), "should be != 0"); - // This method uses pcmpestri inxtruction with bound registers + // This method uses pcmpestri instruction with bound registers // inputs: // xmm - substring // rax - substring length (elements count) @@ -6644,7 +6644,6 @@ void MacroAssembler::string_compare(Register str1, Register str2, // start from first character again because it has aligned address. int stride2 = 16; int adr_stride = stride << scale; - int adr_stride2 = stride2 << scale; assert(result == rax && cnt2 == rdx && cnt1 == rcx, "pcmpestri"); // rax and rdx are used by pcmpestri as elements counters @@ -6743,7 +6742,7 @@ void MacroAssembler::string_compare(Register str1, Register str2, // inputs: // vec1- substring // rax - negative string length (elements count) - // mem - scaned string + // mem - scanned string // rdx - string length (elements count) // pcmpmask - cmp mode: 11000 (string compare with negated result) // + 00 (unsigned bytes) or + 01 (unsigned shorts) diff --git a/hotspot/src/share/vm/opto/library_call.cpp b/hotspot/src/share/vm/opto/library_call.cpp index 7c9da72c68e..45eeb9672ff 100644 --- a/hotspot/src/share/vm/opto/library_call.cpp +++ b/hotspot/src/share/vm/opto/library_call.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1999, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1999, 2015, 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 @@ -1351,7 +1351,6 @@ Node* LibraryCallKit::string_indexOf(Node* string_object, ciTypeArray* target_ar Node* cache = __ ConI(cache_i); Node* md2 = __ ConI(md2_i); Node* lastChar = __ ConI(target_array->char_at(target_length - 1)); - Node* targetCount = __ ConI(target_length); Node* targetCountLess1 = __ ConI(target_length - 1); Node* targetOffset = __ ConI(targetOffset_i); Node* sourceEnd = __ SubI(__ AddI(sourceOffset, sourceCount), targetCountLess1); @@ -1408,8 +1407,6 @@ bool LibraryCallKit::inline_string_indexOf() { Node* arg = argument(1); Node* result; - // Disable the use of pcmpestri until it can be guaranteed that - // the load doesn't cross into the uncommited space. if (Matcher::has_match_rule(Op_StrIndexOf) && UseSSE42Intrinsics) { // Generate SSE4.2 version of indexOf @@ -1421,9 +1418,6 @@ bool LibraryCallKit::inline_string_indexOf() { return true; } - ciInstanceKlass* str_klass = env()->String_klass(); - const TypeOopPtr* string_type = TypeOopPtr::make_from_klass(str_klass); - // Make the merge point RegionNode* result_rgn = new RegionNode(4); Node* result_phi = new PhiNode(result_rgn, TypeInt::INT); From 01ec6950020268a375eb5330373e6c0813f3a924 Mon Sep 17 00:00:00 2001 From: Nils Eliasson Date: Thu, 22 Jan 2015 11:05:14 +0100 Subject: [PATCH 07/58] 8069389: CompilerOracle prefix wildcarding is broken for long strings Replace strcpy with memmove Reviewed-by: kvn --- .../src/share/vm/compiler/compilerOracle.cpp | 3 +- .../compiler/oracle/TestCompileCommand.java | 78 +++++++++++++++++++ 2 files changed, 80 insertions(+), 1 deletion(-) create mode 100644 hotspot/test/compiler/oracle/TestCompileCommand.java diff --git a/hotspot/src/share/vm/compiler/compilerOracle.cpp b/hotspot/src/share/vm/compiler/compilerOracle.cpp index a6351be1783..1c90c106e62 100644 --- a/hotspot/src/share/vm/compiler/compilerOracle.cpp +++ b/hotspot/src/share/vm/compiler/compilerOracle.cpp @@ -553,7 +553,8 @@ static MethodMatcher::Mode check_mode(char name[], const char*& error_msg) { int match = MethodMatcher::Exact; while (name[0] == '*') { match |= MethodMatcher::Suffix; - strcpy(name, name + 1); + // Copy remaining string plus NUL to the beginning + memmove(name, name + 1, strlen(name + 1) + 1); } if (strcmp(name, "*") == 0) return MethodMatcher::Any; diff --git a/hotspot/test/compiler/oracle/TestCompileCommand.java b/hotspot/test/compiler/oracle/TestCompileCommand.java new file mode 100644 index 00000000000..e2575ed1fa5 --- /dev/null +++ b/hotspot/test/compiler/oracle/TestCompileCommand.java @@ -0,0 +1,78 @@ +/* + * Copyright (c) 2015, 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. + */ + +import java.io.PrintWriter; +import java.io.File; + +import com.oracle.java.testlibrary.*; + +/* + * @test TestCompileCommand + * @bug 8069389 + * @summary "Regression tests of -XX:CompileCommand" + * @library /testlibrary + * @run main TestCompileCommand + */ + +public class TestCompileCommand { + + private static final String[][] ARGUMENTS = { + { + "-XX:CompileCommand=print,*01234567890123456789012345678901234567890123456789,*0123456789012345678901234567890123456789", + "-version" + } + }; + + private static final String[][] OUTPUTS = { + { + "print *01234567890123456789012345678901234567890123456789.*0123456789012345678901234567890123456789" + } + }; + + private static void verifyValidOption(String[] arguments, String[] expected_outputs) throws Exception { + ProcessBuilder pb; + OutputAnalyzer out; + + pb = ProcessTools.createJavaProcessBuilder(arguments); + out = new OutputAnalyzer(pb.start()); + + for (String expected_output : expected_outputs) { + out.shouldContain(expected_output); + } + + out.shouldNotContain("CompileCommand: An error occured during parsing"); + out.shouldHaveExitValue(0); + } + + public static void main(String[] args) throws Exception { + + if (ARGUMENTS.length != OUTPUTS.length) { + throw new RuntimeException("Test is set up incorrectly: length of arguments and expected outputs for type (1) options does not match."); + } + + // Check if type (1) options are parsed correctly + for (int i = 0; i < ARGUMENTS.length; i++) { + verifyValidOption(ARGUMENTS[i], OUTPUTS[i]); + } + } +} From a43e328b4a834550d93810ad38d1ace47fad13d5 Mon Sep 17 00:00:00 2001 From: Nils Eliasson Date: Thu, 22 Jan 2015 11:23:13 +0100 Subject: [PATCH 08/58] 8069035: compiler/oracle/CheckCompileCommandOption.java nightly failure Fixed whitespace handling and added test cases Reviewed-by: kvn, anoll, zmajo --- .../src/share/vm/compiler/compilerOracle.cpp | 20 +-- .../oracle/CheckCompileCommandOption.java | 129 +++++++++++------- hotspot/test/compiler/oracle/command1.txt | 12 ++ hotspot/test/compiler/oracle/command2.txt | 7 + 4 files changed, 110 insertions(+), 58 deletions(-) create mode 100644 hotspot/test/compiler/oracle/command1.txt create mode 100644 hotspot/test/compiler/oracle/command2.txt diff --git a/hotspot/src/share/vm/compiler/compilerOracle.cpp b/hotspot/src/share/vm/compiler/compilerOracle.cpp index 1c90c106e62..8299da22f0f 100644 --- a/hotspot/src/share/vm/compiler/compilerOracle.cpp +++ b/hotspot/src/share/vm/compiler/compilerOracle.cpp @@ -690,6 +690,13 @@ static MethodMatcher* scan_flag_and_value(const char* type, const char* line, in return NULL; } +int skip_whitespace(char* line) { + // Skip any leading spaces + int whitespace_read = 0; + sscanf(line, "%*[ \t]%n", &whitespace_read); + return whitespace_read; +} + void CompilerOracle::parse_from_line(char* line) { if (line[0] == '\0') return; if (line[0] == '#') return; @@ -756,15 +763,9 @@ void CompilerOracle::parse_from_line(char* line) { line += bytes_read; - // Skip any leading spaces before signature - int whitespace_read = 0; - sscanf(line, "%*[ \t]%n", &whitespace_read); - if (whitespace_read > 0) { - line += whitespace_read; - } - // there might be a signature following the method. // signatures always begin with ( so match that by hand + line += skip_whitespace(line); if (1 == sscanf(line, "(%254[[);/" RANGEBASE "]%n", sig + 1, &bytes_read)) { sig[0] = '('; line += bytes_read; @@ -787,7 +788,9 @@ void CompilerOracle::parse_from_line(char* line) { // // For future extensions: extend scan_flag_and_value() char option[256]; // stores flag for Type (1) and type of Type (2) - while (sscanf(line, "%*[ \t]%255[a-zA-Z0-9]%n", option, &bytes_read) == 1) { + + line += skip_whitespace(line); + while (sscanf(line, "%255[a-zA-Z0-9]%n", option, &bytes_read) == 1) { if (match != NULL && !_quiet) { // Print out the last match added ttyLocker ttyl; @@ -817,6 +820,7 @@ void CompilerOracle::parse_from_line(char* line) { // Type (1) option match = add_option_string(c_name, c_match, m_name, m_match, signature, option, true); } + line += skip_whitespace(line); } // while( } else { match = add_predicate(command, c_name, c_match, m_name, m_match, signature); diff --git a/hotspot/test/compiler/oracle/CheckCompileCommandOption.java b/hotspot/test/compiler/oracle/CheckCompileCommandOption.java index 8248582d38c..6d69b5e7dd4 100644 --- a/hotspot/test/compiler/oracle/CheckCompileCommandOption.java +++ b/hotspot/test/compiler/oracle/CheckCompileCommandOption.java @@ -21,12 +21,15 @@ * questions. */ +import java.io.PrintWriter; +import java.io.File; + import com.oracle.java.testlibrary.*; /* * @test CheckCompileCommandOption - * @bug 8055286 8056964 8059847 - * @summary "Checks parsing of -XX:+CompileCommand=option" + * @bug 8055286 8056964 8059847 8069035 + * @summary "Checks parsing of -XX:CompileCommand=option" * @library /testlibrary * @run main CheckCompileCommandOption */ @@ -45,38 +48,69 @@ public class CheckCompileCommandOption { // have the the following types: intx, uintx, bool, ccstr, // ccstrlist, and double. + private static final String[][] FILE_ARGUMENTS = { + { + "-XX:CompileCommandFile=" + new File(System.getProperty("test.src", "."), "command1.txt"), + "-version" + }, + { + "-XX:CompileCommandFile=" + new File(System.getProperty("test.src", "."), "command2.txt"), + "-version" + } + }; + + private static final String[][] FILE_EXPECTED_OUTPUT = { + { + "CompileCommand: option com/oracle/Test.test bool MyBoolOption1 = true", + "CompileCommand: option com/oracle/Test.test bool MyBoolOption2 = true", + "CompileCommand: option com/oracle/Test.test bool MyBoolOption3 = true", + "CompileCommand: option com/oracle/Test.test bool MyBoolOption4 = true", + "CompileCommand: option com/oracle/Test.test bool MyBoolOption5 = true", + "CompileCommand: option com/oracle/Test.test bool MyBoolOption6 = true", + "CompileCommand: option com/oracle/Test.test bool MyBoolOption7 = true", + "CompileCommand: option com/oracle/Test.test bool MyBoolOption8 = true", + "CompileCommand: option com/oracle/Test.test(I) bool MyBoolOption9 = true", + "CompileCommand: option com/oracle/Test.test(I) bool MyBoolOption10 = true", + "CompileCommand: option com/oracle/Test.test(I) bool MyBoolOption11 = true", + "CompileCommand: option com/oracle/Test.test(I) bool MyBoolOption12 = true", + "CompileCommand: option com/oracle/Test.test(I) bool MyBoolOption13 = true", + "CompileCommand: option com/oracle/Test.test(I) bool MyBoolOption14 = true", + "CompileCommand: option com/oracle/Test.test(I) bool MyBoolOption15 = true", + "CompileCommand: option com/oracle/Test.test(I) bool MyBoolOption16 = true" + }, + { + "CompileCommand: option Test.test const char* MyListOption = '_foo _bar'", + "CompileCommand: option Test.test const char* MyStrOption = '_foo'", + "CompileCommand: option Test.test bool MyBoolOption = false", + "CompileCommand: option Test.test intx MyIntxOption = -1", + "CompileCommand: option Test.test uintx MyUintxOption = 1", + "CompileCommand: option Test.test bool MyFlag = true", + "CompileCommand: option Test.test double MyDoubleOption = 1.123000" + } + }; + private static final String[][] TYPE_1_ARGUMENTS = { { "-XX:CompileCommand=option,com/oracle/Test.test,MyBoolOption1", "-XX:CompileCommand=option,com/oracle/Test,test,MyBoolOption2", "-XX:CompileCommand=option,com.oracle.Test::test,MyBoolOption3", "-XX:CompileCommand=option,com/oracle/Test::test,MyBoolOption4", - "-version" - }, - { - "-XX:CompileCommand=option,com/oracle/Test.test,MyBoolOption1,MyBoolOption2", - "-version" - }, - { - "-XX:CompileCommand=option,com/oracle/Test,test,MyBoolOption1,MyBoolOption2", + "-XX:CompileCommand=option,com/oracle/Test.test,MyBoolOption5,MyBoolOption6", + "-XX:CompileCommand=option,com/oracle/Test,test,MyBoolOption7,MyBoolOption8", "-version" } }; private static final String[][] TYPE_1_EXPECTED_OUTPUTS = { { - "CompilerOracle: option com/oracle/Test.test bool MyBoolOption1 = true", - "CompilerOracle: option com/oracle/Test.test bool MyBoolOption2 = true", - "CompilerOracle: option com/oracle/Test.test bool MyBoolOption3 = true", - "CompilerOracle: option com/oracle/Test.test bool MyBoolOption4 = true" - }, - { - "CompilerOracle: option com/oracle/Test.test bool MyBoolOption1 = true", - "CompilerOracle: option com/oracle/Test.test bool MyBoolOption2 = true", - }, - { - "CompilerOracle: option com/oracle/Test.test bool MyBoolOption1 = true", - "CompilerOracle: option com/oracle/Test.test bool MyBoolOption2 = true", + "CompileCommand: option com/oracle/Test.test bool MyBoolOption1 = true", + "CompileCommand: option com/oracle/Test.test bool MyBoolOption2 = true", + "CompileCommand: option com/oracle/Test.test bool MyBoolOption3 = true", + "CompileCommand: option com/oracle/Test.test bool MyBoolOption4 = true", + "CompileCommand: option com/oracle/Test.test bool MyBoolOption5 = true", + "CompileCommand: option com/oracle/Test.test bool MyBoolOption6 = true", + "CompileCommand: option com/oracle/Test.test bool MyBoolOption7 = true", + "CompileCommand: option com/oracle/Test.test bool MyBoolOption8 = true" } }; @@ -88,38 +122,28 @@ public class CheckCompileCommandOption { "-XX:CompileCommand=option,Test::test,intx,MyIntxOption,-1", "-XX:CompileCommand=option,Test::test,uintx,MyUintxOption,1", "-XX:CompileCommand=option,Test::test,MyFlag", - "-XX:CompileCommand=option,Test::test,double,MyDoubleOption,1.123", - "-version" - }, - { - "-XX:CompileCommand=option,Test.test,double,MyDoubleOption,1.123", - "-version" - }, - { - "-XX:CompileCommand=option,Test::test,bool,MyBoolOption,false,intx,MyIntxOption,-1,uintx,MyUintxOption,1,MyFlag,double,MyDoubleOption,1.123", + "-XX:CompileCommand=option,Test::test,double,MyDoubleOption1,1.123", + "-XX:CompileCommand=option,Test.test,double,MyDoubleOption2,1.123", + "-XX:CompileCommand=option,Test::test,bool,MyBoolOptionX,false,intx,MyIntxOptionX,-1,uintx,MyUintxOptionX,1,MyFlagX,double,MyDoubleOptionX,1.123", "-version" } }; private static final String[][] TYPE_2_EXPECTED_OUTPUTS = { { - "CompilerOracle: option Test.test const char* MyListOption = '_foo _bar'", - "CompilerOracle: option Test.test const char* MyStrOption = '_foo'", - "CompilerOracle: option Test.test bool MyBoolOption = false", - "CompilerOracle: option Test.test intx MyIntxOption = -1", - "CompilerOracle: option Test.test uintx MyUintxOption = 1", - "CompilerOracle: option Test.test bool MyFlag = true", - "CompilerOracle: option Test.test double MyDoubleOption = 1.123000" - }, - { - "CompilerOracle: option Test.test double MyDoubleOption = 1.123000" - }, - { - "CompilerOracle: option Test.test bool MyBoolOption = false", - "CompilerOracle: option Test.test intx MyIntxOption = -1", - "CompilerOracle: option Test.test uintx MyUintxOption = 1", - "CompilerOracle: option Test.test bool MyFlag = true", - "CompilerOracle: option Test.test double MyDoubleOption = 1.123000", + "CompileCommand: option Test.test const char* MyListOption = '_foo _bar'", + "CompileCommand: option Test.test const char* MyStrOption = '_foo'", + "CompileCommand: option Test.test bool MyBoolOption = false", + "CompileCommand: option Test.test intx MyIntxOption = -1", + "CompileCommand: option Test.test uintx MyUintxOption = 1", + "CompileCommand: option Test.test bool MyFlag = true", + "CompileCommand: option Test.test double MyDoubleOption1 = 1.123000", + "CompileCommand: option Test.test double MyDoubleOption2 = 1.123000", + "CompileCommand: option Test.test bool MyBoolOptionX = false", + "CompileCommand: option Test.test intx MyIntxOptionX = -1", + "CompileCommand: option Test.test uintx MyUintxOptionX = 1", + "CompileCommand: option Test.test bool MyFlagX = true", + "CompileCommand: option Test.test double MyDoubleOptionX = 1.123000", } }; @@ -172,7 +196,7 @@ public class CheckCompileCommandOption { out.shouldContain(expected_output); } - out.shouldNotContain("CompileCommand: unrecognized line"); + out.shouldNotContain("CompileCommand: An error occured during parsing"); out.shouldHaveExitValue(0); } @@ -183,7 +207,7 @@ public class CheckCompileCommandOption { pb = ProcessTools.createJavaProcessBuilder(arguments); out = new OutputAnalyzer(pb.start()); - out.shouldContain("CompileCommand: unrecognized line"); + out.shouldContain("CompileCommand: An error occured during parsing"); out.shouldHaveExitValue(0); } @@ -212,5 +236,10 @@ public class CheckCompileCommandOption { for (String[] arguments: TYPE_2_INVALID_ARGUMENTS) { verifyInvalidOption(arguments); } + + // Check if commands in command file are parsed correctly + for (int i = 0; i < FILE_ARGUMENTS.length; i++) { + verifyValidOption(FILE_ARGUMENTS[i], FILE_EXPECTED_OUTPUT[i]); + } } } diff --git a/hotspot/test/compiler/oracle/command1.txt b/hotspot/test/compiler/oracle/command1.txt new file mode 100644 index 00000000000..aaf7cbdaecf --- /dev/null +++ b/hotspot/test/compiler/oracle/command1.txt @@ -0,0 +1,12 @@ +option,com/oracle/Test.test,MyBoolOption1 +option,com/oracle/Test,test,MyBoolOption2 +option,com.oracle.Test::test,MyBoolOption3 +option,com/oracle/Test::test,MyBoolOption4 +option,com/oracle/Test.test,MyBoolOption5,MyBoolOption6 +option,com/oracle/Test,test,MyBoolOption7,MyBoolOption8 +option,com/oracle/Test.test(I),MyBoolOption9 +option,com/oracle/Test,test,(I),MyBoolOption10 +option,com.oracle.Test::test(I),MyBoolOption11 +option,com/oracle/Test::test(I),MyBoolOption12 +option,com/oracle/Test.test(I),MyBoolOption13,MyBoolOption14 +option,com/oracle/Test,test(I),MyBoolOption15,MyBoolOption16 diff --git a/hotspot/test/compiler/oracle/command2.txt b/hotspot/test/compiler/oracle/command2.txt new file mode 100644 index 00000000000..d6b5279c881 --- /dev/null +++ b/hotspot/test/compiler/oracle/command2.txt @@ -0,0 +1,7 @@ +option,Test::test,ccstrlist,MyListOption,_foo,_bar +option,Test::test,ccstr,MyStrOption,_foo +option,Test::test,bool,MyBoolOption,false +option,Test::test,intx,MyIntxOption,-1 +option,Test::test,uintx,MyUintxOption,1 +option,Test::test,MyFlag +option,Test::test,double,MyDoubleOption,1.123 From 903cbcb4df5331a4c8ea2b3bb4a2df418414634d Mon Sep 17 00:00:00 2001 From: Igor Veresov Date: Thu, 22 Jan 2015 11:25:23 -0800 Subject: [PATCH 09/58] 8071302: assert(!_reg_node[reg_lo] || edge_from_to(_reg_node[reg_lo], def)) failed: after block local Add merge nodes to node to block mapping Reviewed-by: kvn, vlivanov --- hotspot/src/share/vm/opto/output.cpp | 2 +- hotspot/src/share/vm/opto/postaloc.cpp | 1 + 2 files changed, 2 insertions(+), 1 deletion(-) diff --git a/hotspot/src/share/vm/opto/output.cpp b/hotspot/src/share/vm/opto/output.cpp index 6284abde6e2..b0bdb3c16da 100644 --- a/hotspot/src/share/vm/opto/output.cpp +++ b/hotspot/src/share/vm/opto/output.cpp @@ -2475,7 +2475,7 @@ void Scheduling::DoScheduling() { if( iop == Op_Con ) continue; // Do not schedule Top if( iop == Op_Node && // Do not schedule PhiNodes, ProjNodes mach->pipeline() == MachNode::pipeline_class() && - !n->is_SpillCopy() ) // Breakpoints, Prolog, etc + !n->is_SpillCopy() && !n->is_MachMerge() ) // Breakpoints, Prolog, etc continue; break; // Funny loop structure to be sure... } diff --git a/hotspot/src/share/vm/opto/postaloc.cpp b/hotspot/src/share/vm/opto/postaloc.cpp index d6ac1020266..2aba90329f2 100644 --- a/hotspot/src/share/vm/opto/postaloc.cpp +++ b/hotspot/src/share/vm/opto/postaloc.cpp @@ -428,6 +428,7 @@ int PhaseChaitin::possibly_merge_multidef(Node *n, uint k, Block *block, RegToDe // Insert the merge node into the block before the first use. uint use_index = block->find_node(reg2defuse.at(reg).first_use()); block->insert_node(merge, use_index++); + _cfg.map_node_to_block(merge, block); // Let the allocator know about the new node, use the same lrg _lrg_map.extend(merge->_idx, lrg); From 167de7ae02ce4e5338b8d1d5efc4c852ed7d61d6 Mon Sep 17 00:00:00 2001 From: Alexander Kulyakthin Date: Fri, 23 Jan 2015 14:20:52 +0100 Subject: [PATCH 10/58] 8067945: SVC jdk/test/* should be cleaned from JRE layout dependency Reviewed-by: sla --- jdk/test/com/sun/jdi/ImmutableResourceTest.sh | 2 +- jdk/test/com/sun/jdi/JITDebug.sh | 6 +++--- .../sun/jdi/connect/spi/JdiLoadedByCustomLoader.java | 11 +++-------- .../sun/jdi/connect/spi/JdiLoadedByCustomLoader.sh | 2 +- .../HotSpotDiagnosticMXBean/CheckOrigin.java | 6 +----- jdk/test/com/sun/tools/attach/BasicTests.java | 3 +-- jdk/test/com/sun/tools/attach/PermissionTest.java | 4 +--- jdk/test/com/sun/tools/attach/ProviderTest.java | 5 ++--- jdk/test/com/sun/tools/attach/TempDirTest.java | 4 +--- .../jmxremote/bootstrap/CustomLauncherTest.java | 8 +------- .../jmxremote/bootstrap/LocalManagementTest.java | 8 +------- 11 files changed, 16 insertions(+), 43 deletions(-) diff --git a/jdk/test/com/sun/jdi/ImmutableResourceTest.sh b/jdk/test/com/sun/jdi/ImmutableResourceTest.sh index 29c72ed6254..95bc8778b6e 100644 --- a/jdk/test/com/sun/jdi/ImmutableResourceTest.sh +++ b/jdk/test/com/sun/jdi/ImmutableResourceTest.sh @@ -92,7 +92,7 @@ fi # echo "JDK under test is: $TESTJAVA" # -CP="-classpath ${TESTCLASSES}${PATHSEP}${TESTJAVA}/lib/tools.jar" +CP="-classpath ${TESTCLASSES}" # Compile the test class using the classpath we need: # env diff --git a/jdk/test/com/sun/jdi/JITDebug.sh b/jdk/test/com/sun/jdi/JITDebug.sh index c647e1153b5..423fe26521a 100644 --- a/jdk/test/com/sun/jdi/JITDebug.sh +++ b/jdk/test/com/sun/jdi/JITDebug.sh @@ -103,15 +103,15 @@ if [ -z "${TESTJAVA}" ] ; then #if running standalone (no test harness of any kind), compile the #support files and the test case ${TESTJAVA}/bin/javac -d ${TESTCLASSES} \ - -classpath "$TESTJAVA/lib/tools.jar${PATHSEP}${TESTSRC}" \ + -classpath "${TESTSRC}" \ TestScaffold.java VMConnection.java TargetListener.java TargetAdapter.java ${TESTJAVA}/bin/javac -d ${TESTCLASSES} \ - -classpath "$TESTJAVA/lib/tools.jar${PATHSEP}${TESTSRC}" -g \ + -classpath "${TESTSRC}" -g \ JITDebug.java fi echo "JDK under test is: $TESTJAVA" # -CLASSPATH="$TESTJAVA/lib/tools.jar${PATHSEP}${TESTCLASSES}" +CLASSPATH="${TESTCLASSES}" export CLASSPATH CP="-classpath \"${CLASSPATH}\"" # diff --git a/jdk/test/com/sun/jdi/connect/spi/JdiLoadedByCustomLoader.java b/jdk/test/com/sun/jdi/connect/spi/JdiLoadedByCustomLoader.java index c01f54a79f4..11a522c727f 100644 --- a/jdk/test/com/sun/jdi/connect/spi/JdiLoadedByCustomLoader.java +++ b/jdk/test/com/sun/jdi/connect/spi/JdiLoadedByCustomLoader.java @@ -22,9 +22,8 @@ */ /* - * Creates a URLClassLoader from 2 file URLs. The first - * file URL is constructed from the given argument. The - * second is the SDK tools.jar. Once created the test + * Creates a URLClassLoader from a file URL. The file URL + * is constructed from the given argument. Once created the test * attempts to load another test case (ListConnectors) * using the class loader and then it invokes the list() * method. @@ -39,13 +38,9 @@ public class JdiLoadedByCustomLoader { public static void main(String args[]) throws Exception { // create files from given arguments and tools.jar File f1 = new File(args[0]); - String home = System.getProperty("java.home"); - String tools = ".." + File.separatorChar + "lib" + - File.separatorChar + "tools.jar"; - File f2 = (new File(home, tools)).getCanonicalFile(); // create class loader - URL[] urls = { f1.toURL(), f2.toURL() }; + URL[] urls = { f1.toURL() }; URLClassLoader cl = new URLClassLoader(urls); // load ListConnectors using the class loader diff --git a/jdk/test/com/sun/jdi/connect/spi/JdiLoadedByCustomLoader.sh b/jdk/test/com/sun/jdi/connect/spi/JdiLoadedByCustomLoader.sh index 2b104822b17..d508c00c727 100644 --- a/jdk/test/com/sun/jdi/connect/spi/JdiLoadedByCustomLoader.sh +++ b/jdk/test/com/sun/jdi/connect/spi/JdiLoadedByCustomLoader.sh @@ -68,7 +68,7 @@ SOMEOTHERDIR="${TESTCLASSES}"/someotherdir $JAVAC -d "${TESTCLASSES}" "${TESTSRC}"/JdiLoadedByCustomLoader.java mkdir "${SOMEOTHERDIR}" -$JAVAC -d "${SOMEOTHERDIR}" -classpath "${TESTSRC}${PS}${TESTJAVA}/lib/tools.jar" \ +$JAVAC -d "${SOMEOTHERDIR}" -classpath "${TESTSRC}" \ "${TESTSRC}"/ListConnectors.java # Run the test diff --git a/jdk/test/com/sun/management/HotSpotDiagnosticMXBean/CheckOrigin.java b/jdk/test/com/sun/management/HotSpotDiagnosticMXBean/CheckOrigin.java index 378310cf56a..27d80ffa106 100644 --- a/jdk/test/com/sun/management/HotSpotDiagnosticMXBean/CheckOrigin.java +++ b/jdk/test/com/sun/management/HotSpotDiagnosticMXBean/CheckOrigin.java @@ -62,7 +62,7 @@ public class CheckOrigin { "-XX:+UseConcMarkSweepGC", // this will cause UseParNewGC to be FLAG_SET_ERGO "-XX:+PrintGCDetails", "-XX:Flags=" + flagsFile.getAbsolutePath(), - "-cp", System.getProperty("test.class.path") + File.pathSeparator + getToolsJarPath(), + "-cp", System.getProperty("test.class.path"), "CheckOrigin", "-runtests"); @@ -137,8 +137,4 @@ public class CheckOrigin { vm.detach(); } - private static String getToolsJarPath() { - return System.getProperty("java.home") + - "/../lib/tools.jar".replace("/", File.separator); - } } diff --git a/jdk/test/com/sun/tools/attach/BasicTests.java b/jdk/test/com/sun/tools/attach/BasicTests.java index e462cf6463f..e5f4b4bd941 100644 --- a/jdk/test/com/sun/tools/attach/BasicTests.java +++ b/jdk/test/com/sun/tools/attach/BasicTests.java @@ -80,8 +80,7 @@ public class BasicTests { // Need to add jdk/lib/tools.jar to classpath. String classpath = - System.getProperty("test.class.path", "") + File.pathSeparator + - System.getProperty("test.jdk", ".") + sep + "lib" + sep + "tools.jar"; + System.getProperty("test.class.path", ""); String testClassDir = System.getProperty("test.classes", "") + sep; // Argumenta : -classpath cp BasicTests$TestMain pid agent badagent redefineagent diff --git a/jdk/test/com/sun/tools/attach/PermissionTest.java b/jdk/test/com/sun/tools/attach/PermissionTest.java index 471f6fc5953..b171fcea593 100644 --- a/jdk/test/com/sun/tools/attach/PermissionTest.java +++ b/jdk/test/com/sun/tools/attach/PermissionTest.java @@ -71,10 +71,8 @@ public class PermissionTest { private static void runTests(long pid) throws Throwable { final String sep = File.separator; - // Need to add jdk/lib/tools.jar to classpath. String classpath = - System.getProperty("test.class.path", "") + File.pathSeparator + - System.getProperty("test.jdk", ".") + sep + "lib" + sep + "tools.jar"; + System.getProperty("test.class.path", ""); String testSrc = System.getProperty("test.src", "") + sep; // Use a policy that will NOT allow attach. Test will verify exception. diff --git a/jdk/test/com/sun/tools/attach/ProviderTest.java b/jdk/test/com/sun/tools/attach/ProviderTest.java index 883951f444d..146c17f071b 100644 --- a/jdk/test/com/sun/tools/attach/ProviderTest.java +++ b/jdk/test/com/sun/tools/attach/ProviderTest.java @@ -68,11 +68,10 @@ public class ProviderTest { String testClasses = System.getProperty("test.classes", "") + sep; String jdkLib = System.getProperty("test.jdk", ".") + sep + "lib" + sep; - // Need to add SimpleProvider.jar and tools.jar to classpath. + // Need to add SimpleProvider.jar to classpath. String classpath = testClassPath + File.pathSeparator + - testClasses + "SimpleProvider.jar" + File.pathSeparator + - jdkLib + "tools.jar"; + testClasses + "SimpleProvider.jar"; String[] args = { "-classpath", diff --git a/jdk/test/com/sun/tools/attach/TempDirTest.java b/jdk/test/com/sun/tools/attach/TempDirTest.java index 86e3edda742..7aa41dc80d6 100644 --- a/jdk/test/com/sun/tools/attach/TempDirTest.java +++ b/jdk/test/com/sun/tools/attach/TempDirTest.java @@ -120,10 +120,8 @@ public class TempDirTest { private static void launchTests(long pid, Path clientTmpDir) throws Throwable { final String sep = File.separator; - // Need to add jdk/lib/tools.jar to classpath. String classpath = - System.getProperty("test.class.path", "") + File.pathSeparator + - System.getProperty("test.jdk", ".") + sep + "lib" + sep + "tools.jar"; + System.getProperty("test.class.path", ""); String[] tmpDirArg = null; if (clientTmpDir != null) { diff --git a/jdk/test/sun/management/jmxremote/bootstrap/CustomLauncherTest.java b/jdk/test/sun/management/jmxremote/bootstrap/CustomLauncherTest.java index ac0911a52ca..0f2978f9b71 100644 --- a/jdk/test/sun/management/jmxremote/bootstrap/CustomLauncherTest.java +++ b/jdk/test/sun/management/jmxremote/bootstrap/CustomLauncherTest.java @@ -145,13 +145,7 @@ public class CustomLauncherTest { ProcessBuilder client = ProcessTools.createJavaProcessBuilder( "-cp", - TEST_CLASSPATH + - File.pathSeparator + - TEST_JDK + - File.separator + - "lib" + - File.separator + - "tools.jar", + TEST_CLASSPATH, "TestManager", String.valueOf(serverPrc.getPid()), port.get(), diff --git a/jdk/test/sun/management/jmxremote/bootstrap/LocalManagementTest.java b/jdk/test/sun/management/jmxremote/bootstrap/LocalManagementTest.java index 87ab6a9245e..1306f4e31a6 100644 --- a/jdk/test/sun/management/jmxremote/bootstrap/LocalManagementTest.java +++ b/jdk/test/sun/management/jmxremote/bootstrap/LocalManagementTest.java @@ -132,13 +132,7 @@ public class LocalManagementTest { ProcessBuilder client = ProcessTools.createJavaProcessBuilder( "-cp", - TEST_CLASSPATH + - File.pathSeparator + - TEST_JDK + - File.separator + - "lib" + - File.separator + - "tools.jar", + TEST_CLASSPATH, "TestManager", String.valueOf(serverPrc.getPid()), port.get(), From b7eaaba9438262b515893a3f18860131257d1a6e Mon Sep 17 00:00:00 2001 From: Dean Long Date: Fri, 23 Jan 2015 22:39:24 -0500 Subject: [PATCH 11/58] 8031064: build_vm_def.sh not working correctly for new build cross compile Move nm and awk code into vm.make Reviewed-by: dsamersoff, dholmes --- hotspot/make/linux/makefiles/build_vm_def.sh | 16 ---------------- hotspot/make/linux/makefiles/vm.make | 10 ++++++++-- 2 files changed, 8 insertions(+), 18 deletions(-) delete mode 100644 hotspot/make/linux/makefiles/build_vm_def.sh diff --git a/hotspot/make/linux/makefiles/build_vm_def.sh b/hotspot/make/linux/makefiles/build_vm_def.sh deleted file mode 100644 index ea81ff6c22f..00000000000 --- a/hotspot/make/linux/makefiles/build_vm_def.sh +++ /dev/null @@ -1,16 +0,0 @@ -#!/bin/sh - -# If we're cross compiling use that path for nm -if [ "$CROSS_COMPILE_ARCH" != "" ]; then -NM=$ALT_COMPILER_PATH/nm -else -NM=nm -fi - -$NM --defined-only $* \ - | awk '{ - if ($3 ~ /^_ZTV/ || $3 ~ /^gHotSpotVM/) print "\t" $3 ";" - if ($3 ~ /^UseSharedSpaces$/) print "\t" $3 ";" - if ($3 ~ /^_ZN9Arguments17SharedArchivePathE$/) print "\t" $3 ";" - }' \ - | sort -u diff --git a/hotspot/make/linux/makefiles/vm.make b/hotspot/make/linux/makefiles/vm.make index a138a163f20..9a3f8d322b4 100644 --- a/hotspot/make/linux/makefiles/vm.make +++ b/hotspot/make/linux/makefiles/vm.make @@ -1,5 +1,5 @@ # -# Copyright (c) 1999, 2014, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 1999, 2015, 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 @@ -239,8 +239,14 @@ mapfile_reorder : mapfile $(REORDERFILE) rm -f $@ cat $^ > $@ +VMDEF_PAT = ^_ZTV +VMDEF_PAT := ^gHotSpotVM|$(VMDEF_PAT) +VMDEF_PAT := ^UseSharedSpaces$$|$(VMDEF_PAT) +VMDEF_PAT := ^_ZN9Arguments17SharedArchivePathE$$|$(VMDEF_PAT) + vm.def: $(Res_Files) $(Obj_Files) - sh $(GAMMADIR)/make/linux/makefiles/build_vm_def.sh *.o > $@ + $(QUIETLY) $(NM) --defined-only $(Obj_Files) | sort -k3 -u | \ + awk '$$3 ~ /$(VMDEF_PAT)/ { print "\t" $$3 ";" }' > $@ mapfile_ext: rm -f $@ From 04474d85c1337e364edc909747eaa80ce6803248 Mon Sep 17 00:00:00 2001 From: Mattias Tobiasson Date: Mon, 26 Jan 2015 08:52:46 +0100 Subject: [PATCH 12/58] 8044419: TEST_BUG: com/sun/jdi/JdbReadTwiceTest.sh fails when run under root Reviewed-by: dsamersoff, sla --- jdk/test/ProblemList.txt | 3 -- jdk/test/com/sun/jdi/JdbReadTwiceTest.sh | 48 ++++++++++++++---------- 2 files changed, 29 insertions(+), 22 deletions(-) diff --git a/jdk/test/ProblemList.txt b/jdk/test/ProblemList.txt index 4954f772f8a..cbe3f74a044 100644 --- a/jdk/test/ProblemList.txt +++ b/jdk/test/ProblemList.txt @@ -292,9 +292,6 @@ com/sun/jdi/JdbMethodExitTest.sh generic-all # 8043571 com/sun/jdi/RepStep.java generic-all -# 8044419 -com/sun/jdi/JdbReadTwiceTest.sh generic-all - # 8058616 com/sun/jdi/RedefinePop.sh generic-all diff --git a/jdk/test/com/sun/jdi/JdbReadTwiceTest.sh b/jdk/test/com/sun/jdi/JdbReadTwiceTest.sh index b7ca50de096..0c78fe5ec81 100644 --- a/jdk/test/com/sun/jdi/JdbReadTwiceTest.sh +++ b/jdk/test/com/sun/jdi/JdbReadTwiceTest.sh @@ -204,27 +204,37 @@ if [ ! -r c:/ ] ; then clean fi +echo +echo "+++++++++++++++++++++++++++++++++++" +echo "Read an unreadable file - verify the read fails." -if [ ! -r c:/ ] ; then - # Can't make a file unreadable under MKS. - echo - echo "+++++++++++++++++++++++++++++++++++" - echo "Read an unreadable file - verify the read fails." - # If the file exists, we try to read it. The - # read will fail. - mkFiles $HOME/jdb.ini - id > $HOME/jdb.ini - chmod a-r $HOME/jdb.ini - if grep -q "uid=" $HOME/jdb.ini ; then - echo "Unable to make file unreadable, so test will fail. chmod: $HOME/jdb.ini" - if grep -q "uid=0" $HOME/jdb.ini ; then - echo "The test is running as root. Fix infrastructure!" - fi - fi +canMakeUnreadable=No +id > $HOME/jdb.ini +if chmod a-r $HOME/jdb.ini +then + grep -q 'uid=0(' $HOME/jdb.ini 2> /dev/null + case $? in + 0) + echo "Error! Can't make file unreadable running as root" + ;; + 1) + echo "Error! Can't make file unreadable for some other reason (windows?)" + ;; + *) + echo "OK. the file is unreadable" + canMakeUnreadable=Yes + ;; + esac +else + echo "Error! Can't create or chmod file" +fi + +if [ "$canMakeUnreadable" = "Yes" ] +then doit failIfNot 1 "open: $HOME/jdb.ini" - clean fi +clean echo @@ -246,8 +256,8 @@ echo "read $fred" > $here/jdb.ini doit failIfNot 1 "from $fred" - if [ ! -r c:/ ] ; then - # Can't make a file unreadable under MKS + if [ "$canMakeUnreadable" = "Yes" ] + then chmod a-r $fred doit failIfNot 1 "open: $fred" From 1da82181797e15c216199bd6bb92ce3db49b406a Mon Sep 17 00:00:00 2001 From: Zoltan Majo Date: Mon, 26 Jan 2015 10:43:42 +0100 Subject: [PATCH 13/58] 8071312: compiler/arguments/CheckCompileThresholdScaling.java fails Change the way VM handles CompileThresholdScaling==0, update test Reviewed-by: kvn, drchase, fzhinkin --- hotspot/src/share/vm/runtime/arguments.cpp | 10 +++-- .../CheckCompileThresholdScaling.java | 42 ++++++++++++------- 2 files changed, 33 insertions(+), 19 deletions(-) diff --git a/hotspot/src/share/vm/runtime/arguments.cpp b/hotspot/src/share/vm/runtime/arguments.cpp index 0c31c9bc112..6096c98b29f 100644 --- a/hotspot/src/share/vm/runtime/arguments.cpp +++ b/hotspot/src/share/vm/runtime/arguments.cpp @@ -1127,7 +1127,7 @@ static void no_shared_spaces(const char* message) { #endif intx Arguments::scaled_compile_threshold(intx threshold, double scale) { - if (scale == 1.0 || scale < 0.0) { + if (scale == 1.0 || scale <= 0.0) { return threshold; } else { return (intx)(threshold * scale); @@ -1143,7 +1143,7 @@ intx Arguments::scaled_freq_log(intx freq_log, double scale) { // Check value to avoid calculating log2 of 0. if (scale == 0.0) { - return 1; + return freq_log; } intx scaled_freq = scaled_compile_threshold((intx)1 << freq_log, scale); @@ -3479,8 +3479,10 @@ jint Arguments::finalize_vm_init_args(SysClassPath* scp_p, bool scp_assembly_req set_mode_flags(_int); } - if ((TieredCompilation && CompileThresholdScaling == 0) - || (!TieredCompilation && scaled_compile_threshold(CompileThreshold) == 0)) { + // CompileThresholdScaling == 0.0 is same as -Xint: Disable compilation (enable interpreter-only mode), + // but like -Xint, leave compilation thresholds unaffected. + // With tiered compilation disabled, setting CompileThreshold to 0 disables compilation as well. + if ((CompileThresholdScaling == 0.0) || (!TieredCompilation && CompileThreshold == 0)) { set_mode_flags(_int); } diff --git a/hotspot/test/compiler/arguments/CheckCompileThresholdScaling.java b/hotspot/test/compiler/arguments/CheckCompileThresholdScaling.java index 5e611022762..257e70a5bcc 100644 --- a/hotspot/test/compiler/arguments/CheckCompileThresholdScaling.java +++ b/hotspot/test/compiler/arguments/CheckCompileThresholdScaling.java @@ -84,6 +84,13 @@ public class CheckCompileThresholdScaling { "-XX:CompileThresholdScaling=0.75", "-version" }, + { + "-XX:-TieredCompilation", + "-XX:+PrintFlagsFinal", + "-XX:CompileThreshold=1000", + "-XX:CompileThresholdScaling=0.0", + "-version" + }, { "-XX:-TieredCompilation", "-XX:+PrintFlagsFinal", @@ -107,6 +114,11 @@ public class CheckCompileThresholdScaling { "intx CompileThreshold := 750 {pd product}", "double CompileThresholdScaling := 0.750000 {product}" }, + { + "intx CompileThreshold := 1000 {pd product}", + "double CompileThresholdScaling := 0.000000 {product}", + "interpreted mode" + }, { "intx CompileThreshold := 0 {pd product}", "double CompileThresholdScaling := 0.750000 {product}", @@ -295,21 +307,21 @@ public class CheckCompileThresholdScaling { "double CompileThresholdScaling := 2.000000 {product}" }, { - "intx Tier0BackedgeNotifyFreqLog := 0 {product}", - "intx Tier0InvokeNotifyFreqLog := 0 {product}", - "intx Tier23InlineeNotifyFreqLog := 0 {product}", - "intx Tier2BackedgeNotifyFreqLog := 0 {product}", - "intx Tier2InvokeNotifyFreqLog := 0 {product}", - "intx Tier3BackEdgeThreshold := 0 {product}", - "intx Tier3BackedgeNotifyFreqLog := 0 {product}", - "intx Tier3CompileThreshold := 0 {product}", - "intx Tier3InvocationThreshold := 0 {product}", - "intx Tier3InvokeNotifyFreqLog := 0 {product}", - "intx Tier3MinInvocationThreshold := 0 {product}", - "intx Tier4BackEdgeThreshold := 0 {product}", - "intx Tier4CompileThreshold := 0 {product}", - "intx Tier4InvocationThreshold := 0 {product}", - "intx Tier4MinInvocationThreshold := 0 {product}", + "intx Tier0BackedgeNotifyFreqLog := 10 {product}", + "intx Tier0InvokeNotifyFreqLog := 7 {product}", + "intx Tier23InlineeNotifyFreqLog := 20 {product}", + "intx Tier2BackedgeNotifyFreqLog := 14 {product}", + "intx Tier2InvokeNotifyFreqLog := 11 {product}", + "intx Tier3BackEdgeThreshold := 60000 {product}", + "intx Tier3BackedgeNotifyFreqLog := 13 {product}", + "intx Tier3CompileThreshold := 2000 {product}", + "intx Tier3InvocationThreshold := 200 {product}", + "intx Tier3InvokeNotifyFreqLog := 10 {product}", + "intx Tier3MinInvocationThreshold := 100 {product}", + "intx Tier4BackEdgeThreshold := 40000 {product}", + "intx Tier4CompileThreshold := 15000 {product}", + "intx Tier4InvocationThreshold := 5000 {product}", + "intx Tier4MinInvocationThreshold := 600 {product}", "double CompileThresholdScaling := 0.000000 {product}", "interpreted mode" } From 1d41a44b78a75f60d778070da669fdca59f30319 Mon Sep 17 00:00:00 2001 From: Katja Kantserova Date: Mon, 26 Jan 2015 13:50:53 +0100 Subject: [PATCH 14/58] 8071324: com/sun/jdi/ConnectedVMs.java should be quarantined Reviewed-by: sla --- jdk/test/ProblemList.txt | 3 +++ 1 file changed, 3 insertions(+) diff --git a/jdk/test/ProblemList.txt b/jdk/test/ProblemList.txt index cbe3f74a044..c83e162b72a 100644 --- a/jdk/test/ProblemList.txt +++ b/jdk/test/ProblemList.txt @@ -298,6 +298,9 @@ com/sun/jdi/RedefinePop.sh generic-all # 8068645 com/sun/jdi/CatchPatternTest.sh generic-all +# 8069402 +com/sun/jdi/ConnectedVMs.java generic-all + ############################################################################ # jdk_util From 28c5919585620b786a6d3a747c8fa8017ca04be9 Mon Sep 17 00:00:00 2001 From: Pavel Chistyakov Date: Mon, 26 Jan 2015 18:59:09 +0300 Subject: [PATCH 15/58] 8066998: [TESTBUG] compiler/whitebox/ForceNMethodSweepTest.java : sweep shouldn't increase usage Disable background compilation Reviewed-by: kvn --- hotspot/test/compiler/whitebox/ForceNMethodSweepTest.java | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/hotspot/test/compiler/whitebox/ForceNMethodSweepTest.java b/hotspot/test/compiler/whitebox/ForceNMethodSweepTest.java index c08067ec4be..0f90d898625 100644 --- a/hotspot/test/compiler/whitebox/ForceNMethodSweepTest.java +++ b/hotspot/test/compiler/whitebox/ForceNMethodSweepTest.java @@ -34,7 +34,6 @@ import com.oracle.java.testlibrary.InfiniteLoop; /* * @test * @bug 8059624 8064669 - * @ignore 8066998 * @library /testlibrary /../../test/lib * @build ForceNMethodSweepTest * @run main ClassFileInstaller sun.hotspot.WhiteBox @@ -42,7 +41,7 @@ import com.oracle.java.testlibrary.InfiniteLoop; * @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions * -XX:-TieredCompilation -XX:+WhiteBoxAPI * -XX:CompileCommand=compileonly,SimpleTestCase$Helper::* - * ForceNMethodSweepTest + * -XX:-BackgroundCompilation ForceNMethodSweepTest * @summary testing of WB::forceNMethodSweep */ public class ForceNMethodSweepTest extends CompilerWhiteBoxTest { From fb6e6379b050c112dfb8ab31dd0b7819e1eab02f Mon Sep 17 00:00:00 2001 From: Pavel Chistyakov Date: Mon, 26 Jan 2015 19:01:50 +0300 Subject: [PATCH 16/58] 8069125: compiler/codecache/stress tests timeout in nightlies Added timeout between one of test threads iterations, adjusted hole test execution time Reviewed-by: kvn --- .../compiler/codecache/stress/CodeCacheStressRunner.java | 2 +- .../compiler/codecache/stress/OverloadCompileQueueTest.java | 5 +++-- 2 files changed, 4 insertions(+), 3 deletions(-) diff --git a/hotspot/test/compiler/codecache/stress/CodeCacheStressRunner.java b/hotspot/test/compiler/codecache/stress/CodeCacheStressRunner.java index 1a7ae636097..f51f296ee67 100644 --- a/hotspot/test/compiler/codecache/stress/CodeCacheStressRunner.java +++ b/hotspot/test/compiler/codecache/stress/CodeCacheStressRunner.java @@ -36,7 +36,7 @@ public class CodeCacheStressRunner { try { // adjust timeout and substract vm init and exit time long timeout = Utils.adjustTimeout(Utils.DEFAULT_TEST_TIMEOUT); - timeout *= 0.9; + timeout *= 0.8; new TimeLimitedRunner(timeout, 2.0d, this::test).call(); } catch (Exception e) { throw new Error("Exception occurred during test execution", e); diff --git a/hotspot/test/compiler/codecache/stress/OverloadCompileQueueTest.java b/hotspot/test/compiler/codecache/stress/OverloadCompileQueueTest.java index 5f52ef679ea..e759d499468 100644 --- a/hotspot/test/compiler/codecache/stress/OverloadCompileQueueTest.java +++ b/hotspot/test/compiler/codecache/stress/OverloadCompileQueueTest.java @@ -77,7 +77,7 @@ public class OverloadCompileQueueTest implements Runnable { } public OverloadCompileQueueTest() { - Helper.startInfiniteLoopThread(this::lockUnlock); + Helper.startInfiniteLoopThread(this::lockUnlock, 100L); } @Override @@ -99,8 +99,9 @@ public class OverloadCompileQueueTest implements Runnable { private void lockUnlock() { try { + int sleep = Helper.RNG.nextInt(MAX_SLEEP); Helper.WHITE_BOX.lockCompilation(); - Thread.sleep(Helper.RNG.nextInt(MAX_SLEEP)); + Thread.sleep(sleep); } catch (InterruptedException e) { throw new Error("TESTBUG: lockUnlocker thread was unexpectedly interrupted", e); } finally { From 3de0140a8895369bbb22e36254071f0408840005 Mon Sep 17 00:00:00 2001 From: Serguei Spitsyn Date: Mon, 26 Jan 2015 15:46:47 -0800 Subject: [PATCH 17/58] 8068162: jvmtiRedefineClasses.cpp: guarantee(false) failed: OLD and/or OBSOLETE method(s) found Enable the test: java/lang/instrument/IsModifiableClassAgent.java Reviewed-by: coleenp, dcubed --- jdk/test/java/lang/instrument/IsModifiableClassAgent.java | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/jdk/test/java/lang/instrument/IsModifiableClassAgent.java b/jdk/test/java/lang/instrument/IsModifiableClassAgent.java index 31329dfd3a1..bb2130924ba 100644 --- a/jdk/test/java/lang/instrument/IsModifiableClassAgent.java +++ b/jdk/test/java/lang/instrument/IsModifiableClassAgent.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2005, 2007, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2005, 2015, 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 @@ -23,7 +23,6 @@ /** * @test - * @ignore JDK-8068162 * @bug 6331574 * @summary test isModifiableClass * @author Robert Field, Sun Microsystems From 4326ee324a9ee99894993368c1dc9ce748ab44c2 Mon Sep 17 00:00:00 2001 From: Serguei Spitsyn Date: Mon, 26 Jan 2015 15:51:28 -0800 Subject: [PATCH 18/58] 8068162: jvmtiRedefineClasses.cpp: guarantee(false) failed: OLD and/or OBSOLETE method(s) found Adjust Unsafe methods in the itable/vtable if Unsafe is redefined Reviewed-by: coleenp, dcubed --- hotspot/src/share/vm/memory/universe.cpp | 9 ++++++--- hotspot/src/share/vm/memory/universe.hpp | 6 ++---- hotspot/src/share/vm/prims/jvmtiRedefineClasses.cpp | 4 ++++ 3 files changed, 12 insertions(+), 7 deletions(-) diff --git a/hotspot/src/share/vm/memory/universe.cpp b/hotspot/src/share/vm/memory/universe.cpp index f536793d41a..03ae80e41d8 100644 --- a/hotspot/src/share/vm/memory/universe.cpp +++ b/hotspot/src/share/vm/memory/universe.cpp @@ -115,6 +115,7 @@ oop Universe::_the_min_jint_string = NULL; LatestMethodCache* Universe::_finalizer_register_cache = NULL; LatestMethodCache* Universe::_loader_addClass_cache = NULL; LatestMethodCache* Universe::_pd_implies_cache = NULL; +LatestMethodCache* Universe::_throw_illegal_access_error_cache = NULL; oop Universe::_out_of_memory_error_java_heap = NULL; oop Universe::_out_of_memory_error_metaspace = NULL; oop Universe::_out_of_memory_error_class_metaspace = NULL; @@ -130,7 +131,6 @@ oop Universe::_virtual_machine_error_instance = NULL; oop Universe::_vm_exception = NULL; oop Universe::_allocation_context_notification_obj = NULL; -Method* Universe::_throw_illegal_access_error = NULL; Array* Universe::_the_empty_int_array = NULL; Array* Universe::_the_empty_short_array = NULL; Array* Universe::_the_empty_klass_array = NULL; @@ -236,6 +236,7 @@ void Universe::serialize(SerializeClosure* f, bool do_all) { _finalizer_register_cache->serialize(f); _loader_addClass_cache->serialize(f); _pd_implies_cache->serialize(f); + _throw_illegal_access_error_cache->serialize(f); } void Universe::check_alignment(uintx size, uintx alignment, const char* name) { @@ -664,6 +665,7 @@ jint universe_init() { Universe::_finalizer_register_cache = new LatestMethodCache(); Universe::_loader_addClass_cache = new LatestMethodCache(); Universe::_pd_implies_cache = new LatestMethodCache(); + Universe::_throw_illegal_access_error_cache = new LatestMethodCache(); if (UseSharedSpaces) { // Read the data structures supporting the shared spaces (shared @@ -1016,7 +1018,8 @@ bool universe_post_init() { tty->print_cr("Unable to link/verify Unsafe.throwIllegalAccessError method"); return false; // initialization failed (cannot throw exception yet) } - Universe::_throw_illegal_access_error = m; + Universe::_throw_illegal_access_error_cache->init( + SystemDictionary::misc_Unsafe_klass(), m); // Setup method for registering loaded classes in class loader vector InstanceKlass::cast(SystemDictionary::ClassLoader_klass())->link_class(CHECK_false); @@ -1042,7 +1045,7 @@ bool universe_post_init() { return false; // initialization failed } Universe::_pd_implies_cache->init( - SystemDictionary::ProtectionDomain_klass(), m);; + SystemDictionary::ProtectionDomain_klass(), m); } // This needs to be done before the first scavenge/gc, since diff --git a/hotspot/src/share/vm/memory/universe.hpp b/hotspot/src/share/vm/memory/universe.hpp index 0861fba3f0e..04074bbfcdc 100644 --- a/hotspot/src/share/vm/memory/universe.hpp +++ b/hotspot/src/share/vm/memory/universe.hpp @@ -148,8 +148,7 @@ class Universe: AllStatic { static LatestMethodCache* _finalizer_register_cache; // static method for registering finalizable objects static LatestMethodCache* _loader_addClass_cache; // method for registering loaded classes in class loader vector static LatestMethodCache* _pd_implies_cache; // method for checking protection domain attributes - - static Method* _throw_illegal_access_error; + static LatestMethodCache* _throw_illegal_access_error_cache; // Unsafe.throwIllegalAccessError() method // preallocated error objects (no backtrace) static oop _out_of_memory_error_java_heap; @@ -305,6 +304,7 @@ class Universe: AllStatic { static Method* loader_addClass_method() { return _loader_addClass_cache->get_method(); } static Method* protection_domain_implies_method() { return _pd_implies_cache->get_method(); } + static Method* throw_illegal_access_error() { return _throw_illegal_access_error_cache->get_method(); } static oop null_ptr_exception_instance() { return _null_ptr_exception_instance; } static oop arithmetic_exception_instance() { return _arithmetic_exception_instance; } @@ -314,8 +314,6 @@ class Universe: AllStatic { static inline oop allocation_context_notification_obj(); static inline void set_allocation_context_notification_obj(oop obj); - static Method* throw_illegal_access_error() { return _throw_illegal_access_error; } - static Array* the_empty_int_array() { return _the_empty_int_array; } static Array* the_empty_short_array() { return _the_empty_short_array; } static Array* the_empty_method_array() { return _the_empty_method_array; } diff --git a/hotspot/src/share/vm/prims/jvmtiRedefineClasses.cpp b/hotspot/src/share/vm/prims/jvmtiRedefineClasses.cpp index 7da07a3875b..b618720382c 100644 --- a/hotspot/src/share/vm/prims/jvmtiRedefineClasses.cpp +++ b/hotspot/src/share/vm/prims/jvmtiRedefineClasses.cpp @@ -3376,7 +3376,9 @@ void VM_RedefineClasses::AdjustCpoolCacheAndVtable::do_klass(Klass* k) { // not yet in the vtable, because the vtable setup is in progress. // This must be done after we adjust the default_methods and // default_vtable_indices for methods already in the vtable. + // If redefining Unsafe, walk all the vtables looking for entries. if (ik->vtable_length() > 0 && (_the_class_oop->is_interface() + || _the_class_oop == SystemDictionary::misc_Unsafe_klass() || ik->is_subtype_of(_the_class_oop))) { // ik->vtable() creates a wrapper object; rm cleans it up ResourceMark rm(_thread); @@ -3396,7 +3398,9 @@ void VM_RedefineClasses::AdjustCpoolCacheAndVtable::do_klass(Klass* k) { // interface, then we have to call adjust_method_entries() for // every InstanceKlass that has an itable since there isn't a // subclass relationship between an interface and an InstanceKlass. + // If redefining Unsafe, walk all the itables looking for entries. if (ik->itable_length() > 0 && (_the_class_oop->is_interface() + || _the_class_oop == SystemDictionary::misc_Unsafe_klass() || ik->is_subclass_of(_the_class_oop))) { // ik->itable() creates a wrapper object; rm cleans it up ResourceMark rm(_thread); From fa47cc3e215fe4bb7531ff1a78f3965e2e84ac9f Mon Sep 17 00:00:00 2001 From: Christian Tornqvist Date: Tue, 27 Jan 2015 05:51:00 -0800 Subject: [PATCH 19/58] 8071530: Update OS detection code to reflect Windows 10 version change Reviewed-by: sla, mgronlun --- hotspot/src/os/windows/vm/os_windows.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/hotspot/src/os/windows/vm/os_windows.cpp b/hotspot/src/os/windows/vm/os_windows.cpp index f8f17d556d0..902548210fe 100644 --- a/hotspot/src/os/windows/vm/os_windows.cpp +++ b/hotspot/src/os/windows/vm/os_windows.cpp @@ -1693,7 +1693,7 @@ void os::win32::print_windows_version(outputStream* st) { } break; - case 6004: + case 10000: if (is_workstation) { st->print("10"); } else { From ad9c86bc97016444d78fa2a44e5bbc758f7835f6 Mon Sep 17 00:00:00 2001 From: Tobias Hartmann Date: Wed, 28 Jan 2015 07:55:27 +0100 Subject: [PATCH 20/58] 8064940: JMH javac performance regressions on solaris-sparcv9 in 9-b34 Aligning code heap sizes to large page size if supported. Reviewed-by: kvn, anoll --- hotspot/src/share/vm/code/codeCache.cpp | 36 ++++++++++--------- hotspot/src/share/vm/code/codeCache.hpp | 3 +- .../printcodecache/PrintCodeCacheRunner.java | 5 ++- 3 files changed, 26 insertions(+), 18 deletions(-) diff --git a/hotspot/src/share/vm/code/codeCache.cpp b/hotspot/src/share/vm/code/codeCache.cpp index 5a546f50b4f..1d43669209e 100644 --- a/hotspot/src/share/vm/code/codeCache.cpp +++ b/hotspot/src/share/vm/code/codeCache.cpp @@ -199,15 +199,10 @@ void CodeCache::initialize_heaps() { } guarantee(NonProfiledCodeHeapSize + ProfiledCodeHeapSize + NonNMethodCodeHeapSize <= ReservedCodeCacheSize, "Size check"); - // Align reserved sizes of CodeHeaps - size_t non_method_size = ReservedCodeSpace::allocation_align_size_up(NonNMethodCodeHeapSize); - size_t profiled_size = ReservedCodeSpace::allocation_align_size_up(ProfiledCodeHeapSize); - size_t non_profiled_size = ReservedCodeSpace::allocation_align_size_up(NonProfiledCodeHeapSize); - - // Compute initial sizes of CodeHeaps - size_t init_non_method_size = MIN2(InitialCodeCacheSize, non_method_size); - size_t init_profiled_size = MIN2(InitialCodeCacheSize, profiled_size); - size_t init_non_profiled_size = MIN2(InitialCodeCacheSize, non_profiled_size); + // Align CodeHeaps + size_t alignment = heap_alignment(); + size_t non_method_size = align_size_up(NonNMethodCodeHeapSize, alignment); + size_t profiled_size = align_size_down(ProfiledCodeHeapSize, alignment); // Reserve one continuous chunk of memory for CodeHeaps and split it into // parts for the individual heaps. The memory layout looks like this: @@ -216,18 +211,27 @@ void CodeCache::initialize_heaps() { // Profiled nmethods // Non-nmethods // ---------- low ------------ - ReservedCodeSpace rs = reserve_heap_memory(non_profiled_size + profiled_size + non_method_size); + ReservedCodeSpace rs = reserve_heap_memory(ReservedCodeCacheSize); ReservedSpace non_method_space = rs.first_part(non_method_size); ReservedSpace rest = rs.last_part(non_method_size); ReservedSpace profiled_space = rest.first_part(profiled_size); ReservedSpace non_profiled_space = rest.last_part(profiled_size); // Non-nmethods (stubs, adapters, ...) - add_heap(non_method_space, "CodeHeap 'non-nmethods'", init_non_method_size, CodeBlobType::NonNMethod); + add_heap(non_method_space, "CodeHeap 'non-nmethods'", CodeBlobType::NonNMethod); // Tier 2 and tier 3 (profiled) methods - add_heap(profiled_space, "CodeHeap 'profiled nmethods'", init_profiled_size, CodeBlobType::MethodProfiled); + add_heap(profiled_space, "CodeHeap 'profiled nmethods'", CodeBlobType::MethodProfiled); // Tier 1 and tier 4 (non-profiled) methods and native methods - add_heap(non_profiled_space, "CodeHeap 'non-profiled nmethods'", init_non_profiled_size, CodeBlobType::MethodNonProfiled); + add_heap(non_profiled_space, "CodeHeap 'non-profiled nmethods'", CodeBlobType::MethodNonProfiled); +} + +size_t CodeCache::heap_alignment() { + // If large page support is enabled, align code heaps according to large + // page size to make sure that code cache is covered by large pages. + const size_t page_size = os::can_execute_large_page_memory() ? + os::page_size_for_region_unaligned(ReservedCodeCacheSize, 8) : + os::vm_page_size(); + return MAX2(page_size, (size_t) os::vm_allocation_granularity()); } ReservedCodeSpace CodeCache::reserve_heap_memory(size_t size) { @@ -284,7 +288,7 @@ const char* CodeCache::get_code_heap_flag_name(int code_blob_type) { return NULL; } -void CodeCache::add_heap(ReservedSpace rs, const char* name, size_t size_initial, int code_blob_type) { +void CodeCache::add_heap(ReservedSpace rs, const char* name, int code_blob_type) { // Check if heap is needed if (!heap_available(code_blob_type)) { return; @@ -295,8 +299,8 @@ void CodeCache::add_heap(ReservedSpace rs, const char* name, size_t size_initial _heaps->append(heap); // Reserve Space + size_t size_initial = MIN2(InitialCodeCacheSize, rs.size()); size_initial = round_to(size_initial, os::vm_page_size()); - if (!heap->reserve(rs, size_initial, CodeCacheSegmentSize)) { vm_exit_during_initialization("Could not reserve enough space for code cache"); } @@ -840,7 +844,7 @@ void CodeCache::initialize() { } else { // Use a single code heap ReservedCodeSpace rs = reserve_heap_memory(ReservedCodeCacheSize); - add_heap(rs, "CodeCache", InitialCodeCacheSize, CodeBlobType::All); + add_heap(rs, "CodeCache", CodeBlobType::All); } // Initialize ICache flush mechanism diff --git a/hotspot/src/share/vm/code/codeCache.hpp b/hotspot/src/share/vm/code/codeCache.hpp index 287c531a4de..d682f90c5b2 100644 --- a/hotspot/src/share/vm/code/codeCache.hpp +++ b/hotspot/src/share/vm/code/codeCache.hpp @@ -98,12 +98,13 @@ class CodeCache : AllStatic { // CodeHeap management static void initialize_heaps(); // Initializes the CodeHeaps // Creates a new heap with the given name and size, containing CodeBlobs of the given type - static void add_heap(ReservedSpace rs, const char* name, size_t size_initial, int code_blob_type); + static void add_heap(ReservedSpace rs, const char* name, int code_blob_type); static CodeHeap* get_code_heap(const CodeBlob* cb); // Returns the CodeHeap for the given CodeBlob static CodeHeap* get_code_heap(int code_blob_type); // Returns the CodeHeap for the given CodeBlobType // Returns the name of the VM option to set the size of the corresponding CodeHeap static const char* get_code_heap_flag_name(int code_blob_type); static bool heap_available(int code_blob_type); // Returns true if an own CodeHeap for the given CodeBlobType is available + static size_t heap_alignment(); // Returns the alignment of the CodeHeaps in bytes static ReservedCodeSpace reserve_heap_memory(size_t size); // Reserves one continuous chunk of memory for the CodeHeaps // Iteration diff --git a/hotspot/test/compiler/codecache/cli/printcodecache/PrintCodeCacheRunner.java b/hotspot/test/compiler/codecache/cli/printcodecache/PrintCodeCacheRunner.java index 74b21f1f19e..c07d3a331eb 100644 --- a/hotspot/test/compiler/codecache/cli/printcodecache/PrintCodeCacheRunner.java +++ b/hotspot/test/compiler/codecache/cli/printcodecache/PrintCodeCacheRunner.java @@ -82,6 +82,9 @@ public class PrintCodeCacheRunner implements CodeCacheCLITestCase.Runner { ExitCode.OK, testCaseDescription.getTestOptions(options, CommandLineOptionTest.prepareBooleanFlag( - "PrintCodeCache", printCodeCache))); + "PrintCodeCache", printCodeCache), + // Do not use large pages to avoid large page + // alignment of code heaps affecting their size. + "-XX:-UseLargePages")); } } From 04d8bf6c334f8ec38d2c93e227af0673a3e78386 Mon Sep 17 00:00:00 2001 From: Katja Kantserova Date: Wed, 28 Jan 2015 09:23:41 +0100 Subject: [PATCH 21/58] 8071582: com/sun/jdi/GetLocalVariables4Test.sh should be quarantined Reviewed-by: sspitsyn, sla --- jdk/test/ProblemList.txt | 3 +++ 1 file changed, 3 insertions(+) diff --git a/jdk/test/ProblemList.txt b/jdk/test/ProblemList.txt index c83e162b72a..7b0a4f7a035 100644 --- a/jdk/test/ProblemList.txt +++ b/jdk/test/ProblemList.txt @@ -301,6 +301,9 @@ com/sun/jdi/CatchPatternTest.sh generic-all # 8069402 com/sun/jdi/ConnectedVMs.java generic-all +# 8067354 +com/sun/jdi/GetLocalVariables4Test.sh windows-all + ############################################################################ # jdk_util From f6d01b369704581825522eaaa4386dbc170c0d18 Mon Sep 17 00:00:00 2001 From: Katja Kantserova Date: Wed, 28 Jan 2015 09:27:49 +0100 Subject: [PATCH 22/58] 8071545: Tests are still excluded while the appropriate bug has been fixed Reviewed-by: sspitsyn, sla --- .../jsr292/RedefineMethodUsedByMultipleMethodHandles.java | 1 - .../test/compiler/profiling/spectrapredefineclass/Launcher.java | 1 - .../profiling/spectrapredefineclass_classloaders/Launcher.java | 1 - 3 files changed, 3 deletions(-) diff --git a/hotspot/test/compiler/jsr292/RedefineMethodUsedByMultipleMethodHandles.java b/hotspot/test/compiler/jsr292/RedefineMethodUsedByMultipleMethodHandles.java index 920431525a8..1695c35558a 100644 --- a/hotspot/test/compiler/jsr292/RedefineMethodUsedByMultipleMethodHandles.java +++ b/hotspot/test/compiler/jsr292/RedefineMethodUsedByMultipleMethodHandles.java @@ -26,7 +26,6 @@ * @bug 8042235 * @summary redefining method used by multiple MethodHandles crashes VM * @compile -XDignore.symbol.file RedefineMethodUsedByMultipleMethodHandles.java - * @ignore 7076820 * @run main RedefineMethodUsedByMultipleMethodHandles */ diff --git a/hotspot/test/compiler/profiling/spectrapredefineclass/Launcher.java b/hotspot/test/compiler/profiling/spectrapredefineclass/Launcher.java index 3c4f6b92bb4..7a645ab0b8f 100644 --- a/hotspot/test/compiler/profiling/spectrapredefineclass/Launcher.java +++ b/hotspot/test/compiler/profiling/spectrapredefineclass/Launcher.java @@ -28,7 +28,6 @@ import com.oracle.java.testlibrary.*; * @bug 8038636 * @library /testlibrary * @build Agent - * @ignore 7076820 * @run main ClassFileInstaller Agent * @run main Launcher * @run main/othervm -XX:-TieredCompilation -XX:-BackgroundCompilation -XX:-UseOnStackReplacement -XX:TypeProfileLevel=222 -XX:ReservedCodeCacheSize=3M Agent diff --git a/hotspot/test/compiler/profiling/spectrapredefineclass_classloaders/Launcher.java b/hotspot/test/compiler/profiling/spectrapredefineclass_classloaders/Launcher.java index 443f11e6c16..3c49cd58ea7 100644 --- a/hotspot/test/compiler/profiling/spectrapredefineclass_classloaders/Launcher.java +++ b/hotspot/test/compiler/profiling/spectrapredefineclass_classloaders/Launcher.java @@ -28,7 +28,6 @@ import com.oracle.java.testlibrary.*; * @bug 8040237 * @library /testlibrary * @build Agent Test A B - * @ignore 7076820 * @run main ClassFileInstaller Agent * @run main Launcher * @run main/othervm -XX:-TieredCompilation -XX:-BackgroundCompilation -XX:-UseOnStackReplacement -XX:TypeProfileLevel=222 -XX:ReservedCodeCacheSize=3M Agent From b5f5424c83e28240a3eb280251ffe604a294ac20 Mon Sep 17 00:00:00 2001 From: Daniel Fuchs Date: Wed, 28 Jan 2015 17:48:59 +0100 Subject: [PATCH 23/58] 8068730: Increase the precision of the implementation of java.time.Clock.systemUTC() Changes the implementation of java.time.Clock.systemUTC() to take advantage of the maximum resolution of the underlying native clock on which System.currentTimeMillis() is based. Reviewed-by: dholmes, rriggs, scolebourne, sla --- .../share/classes/java/time/Clock.java | 64 +++- .../share/classes/java/util/Formatter.java | 10 +- .../java.base/share/classes/sun/misc/VM.java | 32 +- jdk/src/java.base/share/native/include/jvm.h | 5 +- jdk/src/java.base/share/native/libjava/VM.c | 15 +- .../time/tck/java/time/TCKLocalDateTime.java | 5 +- .../java/time/tck/java/time/TCKLocalTime.java | 5 +- .../time/tck/java/time/TCKZonedDateTime.java | 5 +- .../time/test/java/time/TestClock_System.java | 298 +++++++++++++++++- .../time/test/java/util/TestFormatter.java | 41 ++- .../sun/misc/VM/GetNanoTimeAdjustment.java | 249 +++++++++++++++ 11 files changed, 714 insertions(+), 15 deletions(-) create mode 100644 jdk/test/sun/misc/VM/GetNanoTimeAdjustment.java diff --git a/jdk/src/java.base/share/classes/java/time/Clock.java b/jdk/src/java.base/share/classes/java/time/Clock.java index b1127848bed..28dd55f9b8d 100644 --- a/jdk/src/java.base/share/classes/java/time/Clock.java +++ b/jdk/src/java.base/share/classes/java/time/Clock.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2015, 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 @@ -61,12 +61,15 @@ */ package java.time; +import java.io.IOException; +import java.io.ObjectInputStream; import static java.time.LocalTime.NANOS_PER_MINUTE; import static java.time.LocalTime.NANOS_PER_SECOND; import java.io.Serializable; import java.util.Objects; import java.util.TimeZone; +import sun.misc.VM; /** * A clock providing access to the current instant, date and time using a time-zone. @@ -446,10 +449,22 @@ public abstract class Clock { */ static final class SystemClock extends Clock implements Serializable { private static final long serialVersionUID = 6740630888130243051L; + private static final long OFFSET_SEED = + System.currentTimeMillis()/1000 - 1024; // initial offest private final ZoneId zone; + // We don't actually need a volatile here. + // We don't care if offset is set or read concurrently by multiple + // threads - we just need a value which is 'recent enough' - in other + // words something that has been updated at least once in the last + // 2^32 secs (~136 years). And even if we by chance see an invalid + // offset, the worst that can happen is that we will get a -1 value + // from getNanoTimeAdjustment, forcing us to update the offset + // once again. + private transient long offset; SystemClock(ZoneId zone) { this.zone = zone; + this.offset = OFFSET_SEED; } @Override public ZoneId getZone() { @@ -464,11 +479,50 @@ public abstract class Clock { } @Override public long millis() { + // System.currentTimeMillis() and VM.getNanoTimeAdjustment(offset) + // use the same time source - System.currentTimeMillis() simply + // limits the resolution to milliseconds. + // So we take the faster path and call System.currentTimeMillis() + // directly - in order to avoid the performance penalty of + // VM.getNanoTimeAdjustment(offset) which is less efficient. return System.currentTimeMillis(); } @Override public Instant instant() { - return Instant.ofEpochMilli(millis()); + // Take a local copy of offset. offset can be updated concurrently + // by other threads (even if we haven't made it volatile) so we will + // work with a local copy. + long localOffset = offset; + long adjustment = VM.getNanoTimeAdjustment(localOffset); + + if (adjustment == -1) { + // -1 is a sentinel value returned by VM.getNanoTimeAdjustment + // when the offset it is given is too far off the current UTC + // time. In principle, this should not happen unless the + // JVM has run for more than ~136 years (not likely) or + // someone is fiddling with the system time, or the offset is + // by chance at 1ns in the future (very unlikely). + // We can easily recover from all these conditions by bringing + // back the offset in range and retry. + + // bring back the offset in range. We use -1024 to make + // it more unlikely to hit the 1ns in the future condition. + localOffset = System.currentTimeMillis()/1000 - 1024; + + // retry + adjustment = VM.getNanoTimeAdjustment(localOffset); + + if (adjustment == -1) { + // Should not happen: we just recomputed a new offset. + // It should have fixed the issue. + throw new InternalError("Offset " + localOffset + " is not in range"); + } else { + // OK - recovery succeeded. Update the offset for the + // next call... + offset = localOffset; + } + } + return Instant.ofEpochSecond(localOffset, adjustment); } @Override public boolean equals(Object obj) { @@ -485,6 +539,12 @@ public abstract class Clock { public String toString() { return "SystemClock[" + zone + "]"; } + private void readObject(ObjectInputStream is) + throws IOException, ClassNotFoundException { + // ensure that offset is initialized + is.defaultReadObject(); + offset = OFFSET_SEED; + } } //----------------------------------------------------------------------- diff --git a/jdk/src/java.base/share/classes/java/util/Formatter.java b/jdk/src/java.base/share/classes/java/util/Formatter.java index a973ee6889e..3b751b6e86a 100644 --- a/jdk/src/java.base/share/classes/java/util/Formatter.java +++ b/jdk/src/java.base/share/classes/java/util/Formatter.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2003, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2003, 2015, 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 @@ -57,6 +57,7 @@ import java.time.ZoneOffset; import java.time.temporal.ChronoField; import java.time.temporal.TemporalAccessor; import java.time.temporal.TemporalQueries; +import java.time.temporal.UnsupportedTemporalTypeException; import sun.misc.DoubleConsts; import sun.misc.FormattedFloatingDecimal; @@ -4056,7 +4057,12 @@ public final class Formatter implements Closeable, Flushable { break; } case DateTime.NANOSECOND: { // 'N' (000000000 - 999999999) - int i = t.get(ChronoField.MILLI_OF_SECOND) * 1000000; + int i; + try { + i = t.get(ChronoField.NANO_OF_SECOND); + } catch (UnsupportedTemporalTypeException u) { + i = t.get(ChronoField.MILLI_OF_SECOND) * 1000000; + } Flags flags = Flags.ZERO_PAD; sb.append(localizedMagnitude(null, i, flags, 9, l)); break; diff --git a/jdk/src/java.base/share/classes/sun/misc/VM.java b/jdk/src/java.base/share/classes/sun/misc/VM.java index c877e40446f..36bfb606495 100644 --- a/jdk/src/java.base/share/classes/sun/misc/VM.java +++ b/jdk/src/java.base/share/classes/sun/misc/VM.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1996, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1996, 2015, 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 @@ -402,6 +402,36 @@ public class VM { */ public static native long getegid(); + /** + * Get a nanosecond time stamp adjustment in the form of a single long. + * + * This value can be used to create an instant using + * {@link java.time.Instant#ofEpochSecond(long, long) + * java.time.Instant.ofEpochSecond(offsetInSeconds, + * getNanoTimeAdjustment(offsetInSeconds))}. + *

+ * The value returned has the best resolution available to the JVM on + * the current system. + * This is usually down to microseconds - or tenth of microseconds - + * depending on the OS/Hardware and the JVM implementation. + * + * @param offsetInSeconds The offset in seconds from which the nanosecond + * time stamp should be computed. + * + * @apiNote The offset should be recent enough - so that + * {@code offsetInSeconds} is within {@code +/- 2^32} seconds of the + * current UTC time. If the offset is too far off, {@code -1} will be + * returned. As such, {@code -1} must not be considered as a valid + * nano time adjustment, but as an exception value indicating + * that an offset closer to the current time should be used. + * + * @return A nanosecond time stamp adjustment in the form of a single long. + * If the offset is too far off the current time, this method returns -1. + * In that case, the caller should call this method again, passing a + * more accurate offset. + */ + public static native long getNanoTimeAdjustment(long offsetInSeconds); + static { initialize(); } diff --git a/jdk/src/java.base/share/native/include/jvm.h b/jdk/src/java.base/share/native/include/jvm.h index 514378df9b2..7177f2e1d1e 100644 --- a/jdk/src/java.base/share/native/include/jvm.h +++ b/jdk/src/java.base/share/native/include/jvm.h @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -104,6 +104,9 @@ JVM_CurrentTimeMillis(JNIEnv *env, jclass ignored); JNIEXPORT jlong JNICALL JVM_NanoTime(JNIEnv *env, jclass ignored); +JNIEXPORT jlong JNICALL +JVM_GetNanoTimeAdjustment(JNIEnv *env, jclass ignored, jlong offset_secs); + JNIEXPORT void JNICALL JVM_ArrayCopy(JNIEnv *env, jclass ignored, jobject src, jint src_pos, jobject dst, jint dst_pos, jint length); diff --git a/jdk/src/java.base/share/native/libjava/VM.c b/jdk/src/java.base/share/native/libjava/VM.c index 1968a06b01b..14fd1a63487 100644 --- a/jdk/src/java.base/share/native/libjava/VM.c +++ b/jdk/src/java.base/share/native/libjava/VM.c @@ -1,5 +1,5 @@ /* - * Copyright (c) 2004, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2004, 2015, 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 @@ -33,6 +33,11 @@ #include "sun_misc_VM.h" +/* Only register the performance-critical methods */ +static JNINativeMethod methods[] = { + {"getNanoTimeAdjustment", "(J)J", (void *)&JVM_GetNanoTimeAdjustment} +}; + JNIEXPORT jobject JNICALL Java_sun_misc_VM_latestUserDefinedLoader(JNIEnv *env, jclass cls) { return JVM_LatestUserDefinedLoader(env); @@ -49,6 +54,14 @@ Java_sun_misc_VM_initialize(JNIEnv *env, jclass cls) { return; } + // Registers implementations of native methods described in methods[] + // above. + // In particular, registers JVM_GetNanoTimeAdjustment as the implementation + // of the native sun.misc.VM.getNanoTimeAdjustment - avoiding the cost of + // introducing a Java_sun_misc_VM_getNanoTimeAdjustment wrapper + (*env)->RegisterNatives(env, cls, + methods, sizeof(methods)/sizeof(methods[0])); + func_p = (GetJvmVersionInfo_fp) JDK_FindJvmEntry("JVM_GetVersionInfo"); if (func_p != NULL) { jvm_version_info info; diff --git a/jdk/test/java/time/tck/java/time/TCKLocalDateTime.java b/jdk/test/java/time/tck/java/time/TCKLocalDateTime.java index 40ed5347589..51b1163dbca 100644 --- a/jdk/test/java/time/tck/java/time/TCKLocalDateTime.java +++ b/jdk/test/java/time/tck/java/time/TCKLocalDateTime.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2015, 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 @@ -292,7 +292,8 @@ public class TCKLocalDateTime extends AbstractDateTimeTest { expected = LocalDateTime.now(Clock.system(zone)); test = LocalDateTime.now(zone); } - assertEquals(test, expected); + assertEquals(test.truncatedTo(ChronoUnit.SECONDS), + expected.truncatedTo(ChronoUnit.SECONDS)); } //----------------------------------------------------------------------- diff --git a/jdk/test/java/time/tck/java/time/TCKLocalTime.java b/jdk/test/java/time/tck/java/time/TCKLocalTime.java index f79e8fc41f6..88f99ca8c65 100644 --- a/jdk/test/java/time/tck/java/time/TCKLocalTime.java +++ b/jdk/test/java/time/tck/java/time/TCKLocalTime.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2015, 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 @@ -246,7 +246,8 @@ public class TCKLocalTime extends AbstractDateTimeTest { expected = LocalTime.now(Clock.system(zone)); test = LocalTime.now(zone); } - assertEquals(test, expected); + assertEquals(test.truncatedTo(ChronoUnit.SECONDS), + expected.truncatedTo(ChronoUnit.SECONDS)); } //----------------------------------------------------------------------- diff --git a/jdk/test/java/time/tck/java/time/TCKZonedDateTime.java b/jdk/test/java/time/tck/java/time/TCKZonedDateTime.java index fb743e87b14..11ace2decf4 100644 --- a/jdk/test/java/time/tck/java/time/TCKZonedDateTime.java +++ b/jdk/test/java/time/tck/java/time/TCKZonedDateTime.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2015, 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 @@ -260,7 +260,8 @@ public class TCKZonedDateTime extends AbstractDateTimeTest { expected = ZonedDateTime.now(Clock.system(zone)); test = ZonedDateTime.now(zone); } - assertEquals(test, expected); + assertEquals(test.truncatedTo(ChronoUnit.SECONDS), + expected.truncatedTo(ChronoUnit.SECONDS)); } //----------------------------------------------------------------------- diff --git a/jdk/test/java/time/test/java/time/TestClock_System.java b/jdk/test/java/time/test/java/time/TestClock_System.java index 881a76060a1..527008920cc 100644 --- a/jdk/test/java/time/test/java/time/TestClock_System.java +++ b/jdk/test/java/time/test/java/time/TestClock_System.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2015, 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 @@ -62,7 +62,9 @@ package test.java.time; import static org.testng.Assert.assertEquals; import static org.testng.Assert.assertSame; +import java.lang.reflect.Field; import java.time.Clock; +import java.time.Instant; import java.time.ZoneId; import org.testng.annotations.Test; @@ -87,4 +89,298 @@ public class TestClock_System { assertEquals(test.toString(), "SystemClock[Europe/Paris]"); } + //----------------------------------------------------------------------- + + private static String formatTime(String prefix, Instant time) { + return prefix + ": " + time + " - seconds: " + + time.getEpochSecond() + ", nanos: " + + time.getNano(); + } + + public void test_ClockResolution() { + Clock highestUTC = Clock.systemUTC(); + + Instant start = Instant.ofEpochMilli(System.currentTimeMillis()); + + try { + // smoke test + Instant system1 = Instant.ofEpochMilli(System.currentTimeMillis()); + Instant system2 = Instant.ofEpochMilli(System.currentTimeMillis()); + Instant highest1 = highestUTC.instant(); + Instant highest2 = highestUTC.instant(); + System.out.println(formatTime("\nsystemUTC #1 ", system1)); + System.out.println(formatTime("systemUTC #2 ", system2)); + System.out.println(formatTime("highestResolutionUTC #1 ", highest1)); + System.out.println(formatTime("highestResolutionUTC #2 ", highest2)); + + if (system2.isBefore(system1)) { + System.err.println("system2 is before system1!"); + System.err.println(formatTime("\n\tsystem1", system1)); + System.err.println(formatTime("\n\tsystem2", system2)); + throw new RuntimeException("system2 is before system1!" + + formatTime("\n\tsystem1", system1) + + formatTime("\n\tsystem2", system2)); + } + if (highest2.isBefore(highest1)) { + System.err.println("highest2 is before highest1!"); + System.err.println(formatTime("\n\thighest1", system1)); + System.err.println(formatTime("\n\tsystem2", highest2)); + throw new RuntimeException("highest2 is before system1!" + + formatTime("\n\thighest1", system1) + + formatTime("\n\tsystem2", highest2)); + } + + // better test - but depends on implementation details. + // we're not rounding - so highest1 should be greater or equal to + // system1 + system1 = Instant.ofEpochMilli(System.currentTimeMillis()); + highest1 = highestUTC.instant(); + + System.out.println(formatTime("\nsystemUTC ", system1)); + System.out.println(formatTime("highestResolutionUTC ", highest1)); + + if (highest1.isBefore(system1)) { + System.err.println("highest1 is before system1!"); + System.err.println(formatTime("\n\tsystem1", system1)); + System.err.println(formatTime("\n\thighest1", highest1)); + throw new RuntimeException("highest1 is before system1!" + + formatTime("\n\tsystem1", system1) + + formatTime("\n\thighest1", highest1)); + } + + int count=0; + // let's preheat the system a bit: + for (int i = 0; i < 1000 ; i++) { + system1 = Instant.ofEpochMilli(System.currentTimeMillis()); + highest1 = highestUTC.instant(); + final int sysnan = system1.getNano(); + final int nanos = highest1.getNano(); + if ((nanos % 1000000) > 0) { + count++; // we have micro seconds + } + if ((sysnan % 1000000) > 0) { + throw new RuntimeException("Expected only millisecconds " + + "precision for systemUTC, found " + + (sysnan % 1000000) + " remainder."); + } + } + System.out.println("\nNumber of time stamps which had better than" + + " millisecond precision: "+count+"/"+1000); + System.out.println(formatTime("\nsystemUTC ", system1)); + System.out.println(formatTime("highestResolutionUTC ", highest1)); + if (count == 0) { + System.err.println("Something is strange: no microsecond " + + "precision with highestResolutionUTC?"); + throw new RuntimeException("Micro second preccision not reached"); + } + + // check again + if (highest1.isBefore(system1)) { + System.err.println("highest1 is before system1!"); + System.err.println(formatTime("\n\tsystem1", system1)); + System.err.println(formatTime("\n\thighest1", highest1)); + throw new RuntimeException("highest1 is before system1!" + + formatTime("\n\tsystem1", system1) + + formatTime("\n\thighest1", highest1)); + } + + // leap of faith: ensure that highest1 is from within 10 secs of + // system1 + if (highest1.toEpochMilli() != system1.toEpochMilli()) { + long delta = highest1.getEpochSecond() - system1.getEpochSecond(); + if (delta > 10) { + throw new RuntimeException("Unexpected long delay between two clocks (" + + delta + " seconds)" + + formatTime("\n\t system1", system1) + + formatTime("\n\t highest1", highest1)); + + } + } else { + System.out.println("You won the lottery: the two dates are within 1 millisecond!\n"); + } + + } finally { + Instant stop = Instant.ofEpochMilli(System.currentTimeMillis()); + if (start.isAfter(stop)) { + // This should not happen - but can (un)probably be observed + // when switching to summer time, or if another application + // is switching the system date... + System.err.println("Cannot test - date was setback: " + + formatTime("\n\tstarted at", start) + + formatTime("\n\tstopped at", stop) + "\n"); + return; // will prevent exceptions from being propagated. + } + } + } + + static final long MAX_OFFSET = 0x0100000000L; + static final long MIN_OFFSET = -MAX_OFFSET; + + // A helper class to test that SystemClock correctly recomputes + // its offset. + static class SystemClockOffset { + + static final int MILLIS_IN_SECOND = 1000; + static final int NANOS_IN_MILLI = 1000_000; + static final int NANOS_IN_MICRO = 1000; + static final int NANOS_IN_SECOND = 1000_000_000; + + static final boolean verbose = true; + static final Clock systemUTC = Clock.systemUTC(); + static final Field offsetField; + + static { + try { + offsetField = Class.forName("java.time.Clock$SystemClock").getDeclaredField("offset"); + offsetField.setAccessible(true); + } catch (ClassNotFoundException | NoSuchFieldException ex) { + throw new ExceptionInInitializerError(ex); + } + } + + static enum Answer { + + YES, // isOffLimit = YES: we must get -1 + NO, // isOffLimit = NO: we must not not get -1 + MAYBE // isOffLimit = MAYBE: we might get -1 or a valid adjustment. + }; + + static long distance(long one, long two) { + return one > two ? Math.subtractExact(one, two) + : Math.subtractExact(two, one); + } + + static Answer isOffLimits(long before, long after, long offset) { + long relativeDistanceBefore = distance(before, offset); + long relativeDistanceAfter = distance(after, offset); + if (relativeDistanceBefore >= MAX_OFFSET && relativeDistanceAfter >= MAX_OFFSET) { + return Answer.YES; + } + if (relativeDistanceBefore < MAX_OFFSET && relativeDistanceAfter < MAX_OFFSET) { + if (relativeDistanceBefore == 0 || relativeDistanceAfter == 0) { + return Answer.MAYBE; // unlucky case where + } + return Answer.NO; + } + return Answer.MAYBE; + } + + static void testWithOffset(String name, long offset) + throws IllegalAccessException { + testWithOffset(name, offset, systemUTC); + } + + static void testWithOffset(String name, long offset, Clock clock) + throws IllegalAccessException { + offsetField.set(clock, offset); + long beforeMillis = System.currentTimeMillis(); + final Instant instant = clock.instant(); + long afterMillis = System.currentTimeMillis(); + long actualOffset = offsetField.getLong(clock); + long instantMillis = instant.getEpochSecond() * MILLIS_IN_SECOND + + instant.getNano() / NANOS_IN_MILLI; + if (instantMillis < beforeMillis || instantMillis > afterMillis) { + throw new RuntimeException(name + + ": Invalid instant: " + instant + + " (~" + instantMillis + "ms)" + + " when time in millis is in [" + + beforeMillis + ", " + afterMillis + + "] and offset in seconds is " + offset); + } + Answer isOffLimits = isOffLimits(beforeMillis / MILLIS_IN_SECOND, + afterMillis / MILLIS_IN_SECOND, offset); + switch (isOffLimits) { + case YES: + if (actualOffset == offset) { + throw new RuntimeException(name + + ": offset was offlimit but was not recomputed " + + " when time in millis is in [" + + beforeMillis + ", " + afterMillis + + "] and offset in seconds was " + offset); + } + break; + case NO: + if (actualOffset != offset) { + throw new RuntimeException(name + + ": offset was not offlimit but was recomputed."); + } + break; + default: + break; + } + if (distance(actualOffset, instant.getEpochSecond()) >= MAX_OFFSET) { + throw new RuntimeException(name + ": Actual offset is too far off:" + + " offset=" + actualOffset + + "instant.seconds=" + instant.getEpochSecond()); + } + long adjustment = (instant.getEpochSecond() - actualOffset) * NANOS_IN_SECOND + + instant.getNano(); + validateAdjustment(name, actualOffset, beforeMillis, afterMillis, adjustment); + } + + static void validateAdjustment(String name, long offset, long beforeMillis, + long afterMillis, long adjustment) { + System.out.println("Validating adjustment: " + adjustment); + long expectedMax = distance(offset, beforeMillis / MILLIS_IN_SECOND) + * NANOS_IN_SECOND + + (beforeMillis % MILLIS_IN_SECOND) * NANOS_IN_MILLI + + (afterMillis - beforeMillis + 1) * NANOS_IN_MILLI; + long absoluteAdjustment = distance(0, adjustment); + if (absoluteAdjustment > expectedMax) { + long adjSec = absoluteAdjustment / NANOS_IN_SECOND; + long adjMil = (absoluteAdjustment % NANOS_IN_SECOND) / NANOS_IN_MILLI; + long adjMic = (absoluteAdjustment % NANOS_IN_MILLI) / NANOS_IN_MICRO; + long adjNan = (absoluteAdjustment % NANOS_IN_MICRO); + long expSec = expectedMax / NANOS_IN_SECOND; + long expMil = (expectedMax % NANOS_IN_SECOND) / NANOS_IN_MILLI; + long expMic = (expectedMax % NANOS_IN_MILLI) / NANOS_IN_MICRO; + long expNan = (expectedMax % NANOS_IN_MICRO); + System.err.println("Excessive adjustment: " + adjSec + "s, " + + adjMil + "ms, " + adjMic + "mics, " + adjNan + "ns"); + System.err.println("Epected max: " + expSec + "s, " + + expMil + "ms, " + expMic + "mics, " + expNan + "ns"); + + throw new RuntimeException(name + + ": Excessive adjustment: " + adjustment + + " when time in millis is in [" + + beforeMillis + ", " + afterMillis + + "] and offset in seconds is " + offset); + } + } + } + + public void test_OffsetRegular() throws IllegalAccessException { + System.out.println("*** Testing regular cases ***"); + SystemClockOffset.testWithOffset("System.currentTimeMillis()/1000", + System.currentTimeMillis()/1000); + SystemClockOffset.testWithOffset("System.currentTimeMillis()/1000 - 1024", + System.currentTimeMillis()/1000 - 1024); + SystemClockOffset.testWithOffset("System.currentTimeMillis()/1000 + 1024", + System.currentTimeMillis()/1000 + 1024); + } + + public void test_OffsetLimits() throws IllegalAccessException { + System.out.println("*** Testing limits ***"); + SystemClockOffset.testWithOffset("System.currentTimeMillis()/1000 - MAX_OFFSET + 1", + System.currentTimeMillis()/1000 - MAX_OFFSET + 1); + SystemClockOffset.testWithOffset("System.currentTimeMillis()/1000 + MAX_OFFSET - 1", + System.currentTimeMillis()/1000 + MAX_OFFSET - 1); + SystemClockOffset.testWithOffset("System.currentTimeMillis()/1000 - MAX_OFFSET", + System.currentTimeMillis()/1000 - MAX_OFFSET); + SystemClockOffset.testWithOffset("System.currentTimeMillis()/1000 + MAX_OFFSET", + System.currentTimeMillis()/1000 + MAX_OFFSET); + SystemClockOffset.testWithOffset("System.currentTimeMillis()/1000 - MAX_OFFSET - 1024", + System.currentTimeMillis()/1000 - MAX_OFFSET - 1024); + SystemClockOffset.testWithOffset("System.currentTimeMillis()/1000 + MAX_OFFSET + 1024", + System.currentTimeMillis()/1000 + MAX_OFFSET + 1024); + SystemClockOffset.testWithOffset("0", 0); + SystemClockOffset.testWithOffset("-1", -1); + SystemClockOffset.testWithOffset("Integer.MAX_VALUE + System.currentTimeMillis()/1000", + ((long)Integer.MAX_VALUE) + System.currentTimeMillis()/1000); + SystemClockOffset.testWithOffset("System.currentTimeMillis()/1000 - Integer.MIN_VALUE", + System.currentTimeMillis()/1000 - Integer.MIN_VALUE); + SystemClockOffset.testWithOffset("Long.MAX_VALUE", Long.MAX_VALUE); + SystemClockOffset.testWithOffset("System.currentTimeMillis()/1000 - Long.MIN_VALUE", + (Long.MIN_VALUE + System.currentTimeMillis()/1000)*-1); + } } diff --git a/jdk/test/java/time/test/java/util/TestFormatter.java b/jdk/test/java/time/test/java/util/TestFormatter.java index 67a73cee697..5cf2163412c 100644 --- a/jdk/test/java/time/test/java/util/TestFormatter.java +++ b/jdk/test/java/time/test/java/util/TestFormatter.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2015, 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 @@ -38,6 +38,7 @@ import java.time.chrono.Chronology; import java.time.temporal.ChronoField; import java.time.temporal.TemporalQueries; import java.time.temporal.TemporalAccessor; +import java.time.temporal.UnsupportedTemporalTypeException; import java.util.*; @@ -153,6 +154,44 @@ public class TestFormatter { if (verbose) { System.out.printf("%-24s : %s%n", getClassName(dt), out); } + + // expected usually comes from Calendar which only has milliseconds + // precision. So we're going to replace it's N:[nanos] stamp with + // the correct value for nanos. + if ((dt instanceof TemporalAccessor) && expected != null) { + try { + // Get millis & nanos from the dt + final TemporalAccessor ta = (TemporalAccessor) dt; + final int nanos = ta.get(ChronoField.NANO_OF_SECOND); + final int millis = ta.get(ChronoField.MILLI_OF_SECOND); + final String nanstr = String.valueOf(nanos); + final String mistr = String.valueOf(millis); + + // Compute the value of the N:[nanos] field that we expect + // to find in 'out' + final StringBuilder sb = new StringBuilder(); + sb.append("N:["); + for (int i=nanstr.length(); i<9; i++) { + sb.append('0'); + } + sb.append(nanos).append("]"); + + // Compute the truncated value of N:[nanos] field that might + // be in 'expected' when expected was built from Calendar. + final StringBuilder sbm = new StringBuilder(); + sbm.append("N:["); + for (int i=mistr.length(); i<3; i++) { + sbm.append('0'); + } + sbm.append(mistr).append("000000]"); + + // if expected contains the truncated value, replace it with + // the complete value. + expected = expected.replace(sbm.toString(), sb.toString()); + } catch (UnsupportedTemporalTypeException e) { + // nano seconds unsupported - nothing to do... + } + } if (expected != null && !out.equals(expected)) { System.out.printf("%-24s actual: %s%n FAILED; expected: %s%n", getClassName(dt), out, expected); diff --git a/jdk/test/sun/misc/VM/GetNanoTimeAdjustment.java b/jdk/test/sun/misc/VM/GetNanoTimeAdjustment.java new file mode 100644 index 00000000000..bfeb8c0c7f3 --- /dev/null +++ b/jdk/test/sun/misc/VM/GetNanoTimeAdjustment.java @@ -0,0 +1,249 @@ +/* + * Copyright (c) 2015, 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. + */ +import java.util.Objects; +import sun.misc.VM; + +/** + * @test + * @bug 8068730 + * @summary tests that VM.getgetNanoTimeAdjustment() works as expected. + * @run main GetNanoTimeAdjustment + * @author danielfuchs + */ +public class GetNanoTimeAdjustment { + + static final int MILLIS_IN_SECOND = 1000; + static final int NANOS_IN_MILLI = 1000_000; + static final int NANOS_IN_MICRO = 1000; + static final int NANOS_IN_SECOND = 1000_000_000; + + static final boolean verbose = true; + + static final class TestAssertException extends RuntimeException { + TestAssertException(String msg) { super(msg); } + } + + private static void assertEquals(long expected, long received, String msg) { + if (expected != received) { + throw new TestAssertException("Unexpected result for " + msg + + ".\n\texpected: " + expected + + "\n\tactual: " + received); + } else if (verbose) { + System.out.println("Got expected " + msg + ": " + received); + } + } + + private static void assertEquals(Object expected, Object received, String msg) { + if (!Objects.equals(expected, received)) { + throw new TestAssertException("Unexpected result for " + msg + + ".\n\texpected: " + expected + + "\n\tactual: " + received); + } else if (verbose) { + System.out.println("Got expected " + msg + ": " + received); + } + } + + static final long MAX_OFFSET = 0x0100000000L; + static final long MIN_OFFSET = -MAX_OFFSET; + static enum Answer { + YES, // isOffLimit = YES: we must get -1 + NO, // isOffLimit = NO: we must not not get -1 + MAYBE // isOffLimit = MAYBE: we might get -1 or a valid adjustment. + }; + static long distance(long one, long two) { + return one > two ? Math.subtractExact(one, two) + : Math.subtractExact(two, one); + } + + + static Answer isOffLimits(long before, long after, long offset) { + long relativeDistanceBefore = distance(before, offset); + long relativeDistanceAfter = distance(after, offset); + if (relativeDistanceBefore >= MAX_OFFSET && relativeDistanceAfter >= MAX_OFFSET) { + return Answer.YES; + } + if (relativeDistanceBefore < MAX_OFFSET && relativeDistanceAfter < MAX_OFFSET) { + if (relativeDistanceBefore == 0 || relativeDistanceAfter == 0) { + return Answer.MAYBE; // unlucky case where + } + return Answer.NO; + } + return Answer.MAYBE; + } + + static void testWithOffset(String name, long offset) { + System.out.println("Testing with offset: " + name); + long beforeMillis = System.currentTimeMillis(); + long adjustment = VM.getNanoTimeAdjustment(offset); + long afterMillis = System.currentTimeMillis(); + + if (offset >= beforeMillis/MILLIS_IN_SECOND + && offset <= afterMillis/MILLIS_IN_SECOND) { + if (adjustment == -1) { + // it's possible that we have fallen in the unlucky case + // where -1 was the genuine result. let's go backward a bit. + offset = offset - 10; + beforeMillis = System.currentTimeMillis(); + adjustment = VM.getNanoTimeAdjustment(offset); + afterMillis = System.currentTimeMillis(); + if (adjustment == -1) { + throw new RuntimeException(name + ": VM says " + offset + + " secs is too far off, " + + " when time in seconds is in [" + + beforeMillis/MILLIS_IN_SECOND + ", " + + afterMillis/MILLIS_IN_SECOND + + "]"); + } + } + } + + Answer isOffLimit = isOffLimits(beforeMillis/MILLIS_IN_SECOND, + afterMillis/MILLIS_IN_SECOND, offset); + switch (isOffLimit) { + case YES: + if (adjustment != -1) { + throw new RuntimeException(name + + ": VM should have returned -1 for " + + offset + + " when time in seconds is in [" + + beforeMillis/MILLIS_IN_SECOND + ", " + + afterMillis/MILLIS_IN_SECOND + "]"); + } + System.out.println("Got expected exception value: " + adjustment); + break; + case NO: + if (adjustment == -1) { + throw new RuntimeException(name + + "VM says " + offset + + " secs is too far off, " + + " when time in seconds is in [" + + beforeMillis/MILLIS_IN_SECOND + ", " + + afterMillis/MILLIS_IN_SECOND + + "]"); + } + break; + case MAYBE: + System.out.println("Adjustment: " + adjustment); + System.out.println("Can't assert for -1 with offset " + + offset + "(" + name + ")" + + " when time in seconds is in [" + + beforeMillis/MILLIS_IN_SECOND + ", " + + afterMillis/MILLIS_IN_SECOND + + "]"); + // not conclusive + } + + if (isOffLimit == Answer.NO || adjustment != -1) { + System.out.println("Validating adjustment: " + adjustment); + long expectedMax = distance(offset, beforeMillis/MILLIS_IN_SECOND) + * NANOS_IN_SECOND + + (beforeMillis % MILLIS_IN_SECOND) * NANOS_IN_MILLI + + (afterMillis - beforeMillis + 1) * NANOS_IN_MILLI; + long absoluteAdjustment = distance(0, adjustment); + if (absoluteAdjustment > expectedMax) { + long adjSec = absoluteAdjustment / NANOS_IN_SECOND; + long adjMil = (absoluteAdjustment % NANOS_IN_SECOND) / NANOS_IN_MILLI; + long adjMic = (absoluteAdjustment % NANOS_IN_MILLI) / NANOS_IN_MICRO; + long adjNan = (absoluteAdjustment % NANOS_IN_MICRO); + long expSec = expectedMax / NANOS_IN_SECOND; + long expMil = (expectedMax % NANOS_IN_SECOND) / NANOS_IN_MILLI; + long expMic = (expectedMax % NANOS_IN_MILLI) / NANOS_IN_MICRO; + long expNan = (expectedMax % NANOS_IN_MICRO); + System.err.println("Excessive adjustment: " + adjSec + "s, " + + adjMil + "ms, " + adjMic + "mics, " + adjNan + "ns"); + System.err.println("Epected max: " + expSec + "s, " + + expMil + "ms, " + expMic + "mics, " + expNan + "ns"); + + throw new RuntimeException(name + + ": Excessive adjustment: " + adjustment + + " when time in millis is in [" + + beforeMillis + ", " + afterMillis + + "] and offset in seconds is " + offset); + } + } + + } + + static void regular() { + System.out.println("*** Testing regular cases ***"); + final long start = System.currentTimeMillis(); + long offset = start/1000; + long adjustment = VM.getNanoTimeAdjustment(offset); + if (start != offset*1000) { + if (adjustment == -1) { + throw new RuntimeException("VM says " + offset + + " secs is too far off, but time millis is " + + System.currentTimeMillis()); + } + } + if (adjustment == -1) { + offset = System.currentTimeMillis()/1000 - 1024; + adjustment = VM.getNanoTimeAdjustment(offset); + if (adjustment == -1) { + throw new RuntimeException("VM says " + offset + + " secs is too far off, but time millis is " + + System.currentTimeMillis()); + } + } + if (adjustment > (start/1000 - offset + 20)*NANOS_IN_SECOND) { + throw new RuntimeException("Excessive adjustment: " + adjustment); + } + testWithOffset("System.currentTimeMillis()/1000", + System.currentTimeMillis()/1000); + testWithOffset("System.currentTimeMillis()/1000 - 1024", + System.currentTimeMillis()/1000 - 1024); + testWithOffset("System.currentTimeMillis()/1000 + 1024", + System.currentTimeMillis()/1000 + 1024); + } + + static void testLimits() { + System.out.println("*** Testing limits ***"); + testWithOffset("System.currentTimeMillis()/1000 - MAX_OFFSET + 1", + System.currentTimeMillis()/1000 - MAX_OFFSET + 1); + testWithOffset("System.currentTimeMillis()/1000 + MAX_OFFSET - 1", + System.currentTimeMillis()/1000 + MAX_OFFSET - 1); + testWithOffset("System.currentTimeMillis()/1000 - MAX_OFFSET", + System.currentTimeMillis()/1000 - MAX_OFFSET); + testWithOffset("System.currentTimeMillis()/1000 + MAX_OFFSET", + System.currentTimeMillis()/1000 + MAX_OFFSET); + testWithOffset("System.currentTimeMillis()/1000 - MAX_OFFSET - 1024", + System.currentTimeMillis()/1000 - MAX_OFFSET - 1024); + testWithOffset("System.currentTimeMillis()/1000 + MAX_OFFSET + 1024", + System.currentTimeMillis()/1000 + MAX_OFFSET + 1024); + testWithOffset("0", 0); + testWithOffset("-1", -1); + testWithOffset("Integer.MAX_VALUE + System.currentTimeMillis()/1000", + ((long)Integer.MAX_VALUE) + System.currentTimeMillis()/1000); + testWithOffset("System.currentTimeMillis()/1000 - Integer.MIN_VALUE", + System.currentTimeMillis()/1000 - Integer.MIN_VALUE); + testWithOffset("Long.MAX_VALUE", Long.MAX_VALUE); + testWithOffset("System.currentTimeMillis()/1000 - Long.MIN_VALUE", + (Long.MIN_VALUE + System.currentTimeMillis()/1000)*-1); + } + + public static void main(String[] args) throws Exception { + regular(); + testLimits(); + } + +} From 967c448681981568f393270c6e2480c4828b3029 Mon Sep 17 00:00:00 2001 From: Daniel Fuchs Date: Wed, 28 Jan 2015 17:52:48 +0100 Subject: [PATCH 24/58] 8068730: Increase the precision of the implementation of java.time.Clock.systemUTC() Changes the implementation of java.time.Clock.systemUTC() to take advantage of the maximum resolution of the underlying native clock on which System.currentTimeMillis() is based. Reviewed-by: dholmes, rriggs, scolebourne, sla --- hotspot/make/aix/makefiles/mapfile-vers-debug | 3 +- .../make/aix/makefiles/mapfile-vers-product | 3 +- .../bsd/makefiles/mapfile-vers-darwin-debug | 3 +- .../bsd/makefiles/mapfile-vers-darwin-product | 3 +- hotspot/make/bsd/makefiles/mapfile-vers-debug | 3 +- .../make/bsd/makefiles/mapfile-vers-product | 3 +- .../make/linux/makefiles/mapfile-vers-debug | 3 +- .../make/linux/makefiles/mapfile-vers-product | 3 +- hotspot/make/solaris/makefiles/mapfile-vers | 3 +- hotspot/src/os/aix/vm/os_aix.cpp | 11 ++++- hotspot/src/os/bsd/vm/os_bsd.cpp | 10 ++++- hotspot/src/os/linux/vm/os_linux.cpp | 11 ++++- hotspot/src/os/solaris/vm/os_solaris.cpp | 12 +++++- hotspot/src/os/windows/vm/os_windows.cpp | 17 +++++++- hotspot/src/share/vm/prims/jvm.cpp | 42 +++++++++++++++++++ hotspot/src/share/vm/prims/jvm.h | 5 ++- hotspot/src/share/vm/runtime/os.hpp | 4 +- 17 files changed, 122 insertions(+), 17 deletions(-) diff --git a/hotspot/make/aix/makefiles/mapfile-vers-debug b/hotspot/make/aix/makefiles/mapfile-vers-debug index 348c128e434..1b3409a6de0 100644 --- a/hotspot/make/aix/makefiles/mapfile-vers-debug +++ b/hotspot/make/aix/makefiles/mapfile-vers-debug @@ -1,5 +1,5 @@ # -# Copyright (c) 2002, 2014, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 2002, 2015, 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 @@ -132,6 +132,7 @@ SUNWprivate_1.1 { JVM_GetMethodIxSignatureUTF; JVM_GetMethodParameters; JVM_GetMethodTypeAnnotations; + JVM_GetNanoTimeAdjustment; JVM_GetPrimitiveArrayElement; JVM_GetProtectionDomain; JVM_GetStackAccessControlContext; diff --git a/hotspot/make/aix/makefiles/mapfile-vers-product b/hotspot/make/aix/makefiles/mapfile-vers-product index 25bd5bfff7c..966060f7de6 100644 --- a/hotspot/make/aix/makefiles/mapfile-vers-product +++ b/hotspot/make/aix/makefiles/mapfile-vers-product @@ -1,5 +1,5 @@ # -# Copyright (c) 2002, 2014, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 2002, 2015, 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 @@ -130,6 +130,7 @@ SUNWprivate_1.1 { JVM_GetMethodIxNameUTF; JVM_GetMethodIxSignatureUTF; JVM_GetMethodParameters; + JVM_GetNanoTimeAdjustment; JVM_GetPrimitiveArrayElement; JVM_GetProtectionDomain; JVM_GetStackAccessControlContext; diff --git a/hotspot/make/bsd/makefiles/mapfile-vers-darwin-debug b/hotspot/make/bsd/makefiles/mapfile-vers-darwin-debug index ba3e771364c..8779ec5ff1d 100644 --- a/hotspot/make/bsd/makefiles/mapfile-vers-darwin-debug +++ b/hotspot/make/bsd/makefiles/mapfile-vers-darwin-debug @@ -1,5 +1,5 @@ # -# Copyright (c) 2002, 2014, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 2002, 2015, 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 @@ -130,6 +130,7 @@ _JVM_GetMethodIxSignatureUTF _JVM_GetMethodParameters _JVM_GetMethodTypeAnnotations + _JVM_GetNanoTimeAdjustment _JVM_GetPrimitiveArrayElement _JVM_GetProtectionDomain _JVM_GetStackAccessControlContext diff --git a/hotspot/make/bsd/makefiles/mapfile-vers-darwin-product b/hotspot/make/bsd/makefiles/mapfile-vers-darwin-product index ba3e771364c..8779ec5ff1d 100644 --- a/hotspot/make/bsd/makefiles/mapfile-vers-darwin-product +++ b/hotspot/make/bsd/makefiles/mapfile-vers-darwin-product @@ -1,5 +1,5 @@ # -# Copyright (c) 2002, 2014, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 2002, 2015, 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 @@ -130,6 +130,7 @@ _JVM_GetMethodIxSignatureUTF _JVM_GetMethodParameters _JVM_GetMethodTypeAnnotations + _JVM_GetNanoTimeAdjustment _JVM_GetPrimitiveArrayElement _JVM_GetProtectionDomain _JVM_GetStackAccessControlContext diff --git a/hotspot/make/bsd/makefiles/mapfile-vers-debug b/hotspot/make/bsd/makefiles/mapfile-vers-debug index fc05d864ad6..7a5c5430436 100644 --- a/hotspot/make/bsd/makefiles/mapfile-vers-debug +++ b/hotspot/make/bsd/makefiles/mapfile-vers-debug @@ -1,5 +1,5 @@ # -# Copyright (c) 2002, 2014, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 2002, 2015, 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 @@ -132,6 +132,7 @@ SUNWprivate_1.1 { JVM_GetMethodIxSignatureUTF; JVM_GetMethodParameters; JVM_GetMethodTypeAnnotations; + JVM_GetNanoTimeAdjustment; JVM_GetPrimitiveArrayElement; JVM_GetProtectionDomain; JVM_GetStackAccessControlContext; diff --git a/hotspot/make/bsd/makefiles/mapfile-vers-product b/hotspot/make/bsd/makefiles/mapfile-vers-product index fc05d864ad6..7a5c5430436 100644 --- a/hotspot/make/bsd/makefiles/mapfile-vers-product +++ b/hotspot/make/bsd/makefiles/mapfile-vers-product @@ -1,5 +1,5 @@ # -# Copyright (c) 2002, 2014, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 2002, 2015, 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 @@ -132,6 +132,7 @@ SUNWprivate_1.1 { JVM_GetMethodIxSignatureUTF; JVM_GetMethodParameters; JVM_GetMethodTypeAnnotations; + JVM_GetNanoTimeAdjustment; JVM_GetPrimitiveArrayElement; JVM_GetProtectionDomain; JVM_GetStackAccessControlContext; diff --git a/hotspot/make/linux/makefiles/mapfile-vers-debug b/hotspot/make/linux/makefiles/mapfile-vers-debug index fc05d864ad6..7a5c5430436 100644 --- a/hotspot/make/linux/makefiles/mapfile-vers-debug +++ b/hotspot/make/linux/makefiles/mapfile-vers-debug @@ -1,5 +1,5 @@ # -# Copyright (c) 2002, 2014, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 2002, 2015, 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 @@ -132,6 +132,7 @@ SUNWprivate_1.1 { JVM_GetMethodIxSignatureUTF; JVM_GetMethodParameters; JVM_GetMethodTypeAnnotations; + JVM_GetNanoTimeAdjustment; JVM_GetPrimitiveArrayElement; JVM_GetProtectionDomain; JVM_GetStackAccessControlContext; diff --git a/hotspot/make/linux/makefiles/mapfile-vers-product b/hotspot/make/linux/makefiles/mapfile-vers-product index fc05d864ad6..7a5c5430436 100644 --- a/hotspot/make/linux/makefiles/mapfile-vers-product +++ b/hotspot/make/linux/makefiles/mapfile-vers-product @@ -1,5 +1,5 @@ # -# Copyright (c) 2002, 2014, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 2002, 2015, 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 @@ -132,6 +132,7 @@ SUNWprivate_1.1 { JVM_GetMethodIxSignatureUTF; JVM_GetMethodParameters; JVM_GetMethodTypeAnnotations; + JVM_GetNanoTimeAdjustment; JVM_GetPrimitiveArrayElement; JVM_GetProtectionDomain; JVM_GetStackAccessControlContext; diff --git a/hotspot/make/solaris/makefiles/mapfile-vers b/hotspot/make/solaris/makefiles/mapfile-vers index 082b9f0e052..02cf70f3958 100644 --- a/hotspot/make/solaris/makefiles/mapfile-vers +++ b/hotspot/make/solaris/makefiles/mapfile-vers @@ -1,5 +1,5 @@ # -# Copyright (c) 2000, 2014, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 2000, 2015, 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 @@ -132,6 +132,7 @@ SUNWprivate_1.1 { JVM_GetMethodIxSignatureUTF; JVM_GetMethodParameters; JVM_GetMethodTypeAnnotations; + JVM_GetNanoTimeAdjustment; JVM_GetPrimitiveArrayElement; JVM_GetProtectionDomain; JVM_GetStackAccessControlContext; diff --git a/hotspot/src/os/aix/vm/os_aix.cpp b/hotspot/src/os/aix/vm/os_aix.cpp index e3322ca0364..d77a26ffa66 100644 --- a/hotspot/src/os/aix/vm/os_aix.cpp +++ b/hotspot/src/os/aix/vm/os_aix.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1999, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1999, 2015, Oracle and/or its affiliates. All rights reserved. * Copyright 2012, 2014 SAP AG. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * @@ -1115,6 +1115,15 @@ jlong os::javaTimeMillis() { return jlong(time.tv_sec) * 1000 + jlong(time.tv_usec / 1000); } +void os::javaTimeSystemUTC(jlong &seconds, jlong &nanos) { + timeval time; + int status = gettimeofday(&time, NULL); + assert(status != -1, "aix error at gettimeofday()"); + seconds = jlong(time.tv_sec); + nanos = jlong(time.tv_usec) * 1000; +} + + // We need to manually declare mread_real_time, // because IBM didn't provide a prototype in time.h. // (they probably only ever tested in C, not C++) diff --git a/hotspot/src/os/bsd/vm/os_bsd.cpp b/hotspot/src/os/bsd/vm/os_bsd.cpp index cddb9449712..445ff225f8e 100644 --- a/hotspot/src/os/bsd/vm/os_bsd.cpp +++ b/hotspot/src/os/bsd/vm/os_bsd.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1999, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1999, 2015, 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 @@ -984,6 +984,14 @@ jlong os::javaTimeMillis() { return jlong(time.tv_sec) * 1000 + jlong(time.tv_usec / 1000); } +void os::javaTimeSystemUTC(jlong &seconds, jlong &nanos) { + timeval time; + int status = gettimeofday(&time, NULL); + assert(status != -1, "bsd error"); + seconds = jlong(time.tv_sec); + nanos = jlong(time.tv_usec) * 1000; +} + #ifndef __APPLE__ #ifndef CLOCK_MONOTONIC #define CLOCK_MONOTONIC (1) diff --git a/hotspot/src/os/linux/vm/os_linux.cpp b/hotspot/src/os/linux/vm/os_linux.cpp index 4ad556ad388..323f8219ebe 100644 --- a/hotspot/src/os/linux/vm/os_linux.cpp +++ b/hotspot/src/os/linux/vm/os_linux.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1999, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1999, 2015, 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 @@ -1322,6 +1322,15 @@ jlong os::javaTimeMillis() { return jlong(time.tv_sec) * 1000 + jlong(time.tv_usec / 1000); } +void os::javaTimeSystemUTC(jlong &seconds, jlong &nanos) { + timeval time; + int status = gettimeofday(&time, NULL); + assert(status != -1, "linux error"); + seconds = jlong(time.tv_sec); + nanos = jlong(time.tv_usec) * 1000; +} + + #ifndef CLOCK_MONOTONIC #define CLOCK_MONOTONIC (1) #endif diff --git a/hotspot/src/os/solaris/vm/os_solaris.cpp b/hotspot/src/os/solaris/vm/os_solaris.cpp index 8a280b0b5ae..47c0f8a9253 100644 --- a/hotspot/src/os/solaris/vm/os_solaris.cpp +++ b/hotspot/src/os/solaris/vm/os_solaris.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -1475,6 +1475,16 @@ jlong os::javaTimeMillis() { return jlong(t.tv_sec) * 1000 + jlong(t.tv_usec) / 1000; } +void os::javaTimeSystemUTC(jlong &seconds, jlong &nanos) { + timeval t; + if (gettimeofday(&t, NULL) == -1) { + fatal(err_msg("os::javaTimeSystemUTC: gettimeofday (%s)", strerror(errno))); + } + seconds = jlong(t.tv_sec); + nanos = jlong(t.tv_usec) * 1000; +} + + jlong os::javaTimeNanos() { return (jlong)getTimeNanos(); } diff --git a/hotspot/src/os/windows/vm/os_windows.cpp b/hotspot/src/os/windows/vm/os_windows.cpp index 902548210fe..76f6384b7b9 100644 --- a/hotspot/src/os/windows/vm/os_windows.cpp +++ b/hotspot/src/os/windows/vm/os_windows.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -839,6 +839,12 @@ jlong windows_to_java_time(FILETIME wt) { return (a - offset()) / 10000; } +// Returns time ticks in (10th of micro seconds) +jlong windows_to_time_ticks(FILETIME wt) { + jlong a = jlong_from(wt.dwHighDateTime, wt.dwLowDateTime); + return (a - offset()); +} + FILETIME java_to_windows_time(jlong l) { jlong a = (l * 10000) + offset(); FILETIME result; @@ -874,6 +880,15 @@ jlong os::javaTimeMillis() { } } +void os::javaTimeSystemUTC(jlong &seconds, jlong &nanos) { + FILETIME wt; + GetSystemTimeAsFileTime(&wt); + jlong ticks = windows_to_time_ticks(wt); // 10th of micros + jlong secs = jlong(ticks / 10000000); // 10000 * 1000 + seconds = secs; + nanos = jlong(ticks - (secs*10000000)) * 100; +} + jlong os::javaTimeNanos() { if (!win32::_has_performance_count) { return javaTimeMillis() * NANOSECS_PER_MILLISEC; // the best we can do. diff --git a/hotspot/src/share/vm/prims/jvm.cpp b/hotspot/src/share/vm/prims/jvm.cpp index 4f857d6e0af..2d52d7e0723 100644 --- a/hotspot/src/share/vm/prims/jvm.cpp +++ b/hotspot/src/share/vm/prims/jvm.cpp @@ -300,6 +300,48 @@ JVM_LEAF(jlong, JVM_NanoTime(JNIEnv *env, jclass ignored)) return os::javaTimeNanos(); JVM_END +// The function below is actually exposed by sun.misc.VM and not +// java.lang.System, but we choose to keep it here so that it stays next +// to JVM_CurrentTimeMillis and JVM_NanoTime + +const jlong MAX_DIFF_SECS = 0x0100000000; // 2^32 +const jlong MIN_DIFF_SECS = -MAX_DIFF_SECS; // -2^32 + +JVM_LEAF(jlong, JVM_GetNanoTimeAdjustment(JNIEnv *env, jclass ignored, jlong offset_secs)) + JVMWrapper("JVM_GetNanoTimeAdjustment"); + jlong seconds; + jlong nanos; + + os::javaTimeSystemUTC(seconds, nanos); + + // We're going to verify that the result can fit in a long. + // For that we need the difference in seconds between 'seconds' + // and 'offset_secs' to be such that: + // |seconds - offset_secs| < (2^63/10^9) + // We're going to approximate 10^9 ~< 2^30 (1000^3 ~< 1024^3) + // which makes |seconds - offset_secs| < 2^33 + // and we will prefer +/- 2^32 as the maximum acceptable diff + // as 2^32 has a more natural feel than 2^33... + // + // So if |seconds - offset_secs| >= 2^32 - we return a special + // sentinel value (-1) which the caller should take as an + // exception value indicating that the offset given to us is + // too far from range of the current time - leading to too big + // a nano adjustment. The caller is expected to recover by + // computing a more accurate offset and calling this method + // again. (For the record 2^32 secs is ~136 years, so that + // should rarely happen) + // + jlong diff = seconds - offset_secs; + if (diff >= MAX_DIFF_SECS || diff <= MIN_DIFF_SECS) { + return -1; // sentinel value: the offset is too far off the target + } + + // return the adjustment. If you compute a time by adding + // this number of nanoseconds along with the number of seconds + // in the offset you should get the current UTC time. + return (diff * (jlong)1000000000) + nanos; +JVM_END JVM_ENTRY(void, JVM_ArrayCopy(JNIEnv *env, jclass ignored, jobject src, jint src_pos, jobject dst, jint dst_pos, jint length)) diff --git a/hotspot/src/share/vm/prims/jvm.h b/hotspot/src/share/vm/prims/jvm.h index 81faa7197d1..9df1e3bfabf 100644 --- a/hotspot/src/share/vm/prims/jvm.h +++ b/hotspot/src/share/vm/prims/jvm.h @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -131,6 +131,9 @@ JVM_CurrentTimeMillis(JNIEnv *env, jclass ignored); JNIEXPORT jlong JNICALL JVM_NanoTime(JNIEnv *env, jclass ignored); +JNIEXPORT jlong JNICALL +JVM_GetNanoTimeAdjustment(JNIEnv *env, jclass ignored, jlong offset_secs); + JNIEXPORT void JNICALL JVM_ArrayCopy(JNIEnv *env, jclass ignored, jobject src, jint src_pos, jobject dst, jint dst_pos, jint length); diff --git a/hotspot/src/share/vm/runtime/os.hpp b/hotspot/src/share/vm/runtime/os.hpp index ecd2e24a0fa..0815d21f0eb 100644 --- a/hotspot/src/share/vm/runtime/os.hpp +++ b/hotspot/src/share/vm/runtime/os.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -172,10 +172,10 @@ class os: AllStatic { static jlong javaTimeMillis(); static jlong javaTimeNanos(); static void javaTimeNanos_info(jvmtiTimerInfo *info_ptr); + static void javaTimeSystemUTC(jlong &seconds, jlong &nanos); static void run_periodic_checks(); static bool supports_monotonic_clock(); - // Returns the elapsed time in seconds since the vm started. static double elapsedTime(); From 8be0bb5458103b5926cbe0badf4a55c7afff5b03 Mon Sep 17 00:00:00 2001 From: Serguei Spitsyn Date: Thu, 29 Jan 2015 03:11:01 -0800 Subject: [PATCH 25/58] 8008678: JSR 292: constant pool reconstitution must support pseudo strings Keep orig idx from pseudo-string to UTF8, use 2nd lsb CPSlot to mark pseudo-string. Reviewed-by: coleenp, jrose --- hotspot/src/share/vm/oops/constantPool.cpp | 11 +- hotspot/src/share/vm/oops/constantPool.hpp | 34 +++-- .../src/share/vm/prims/methodComparator.cpp | 4 +- .../jvmti/TestLambdaFormRetransformation.java | 138 ++++++++++++++++++ 4 files changed, 162 insertions(+), 25 deletions(-) create mode 100644 hotspot/test/serviceability/jvmti/TestLambdaFormRetransformation.java diff --git a/hotspot/src/share/vm/oops/constantPool.cpp b/hotspot/src/share/vm/oops/constantPool.cpp index 98c72156bc3..0feca59f98c 100644 --- a/hotspot/src/share/vm/oops/constantPool.cpp +++ b/hotspot/src/share/vm/oops/constantPool.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -493,12 +493,7 @@ Symbol* ConstantPool::uncached_klass_ref_at_noresolve(int which) { } char* ConstantPool::string_at_noresolve(int which) { - Symbol* s = unresolved_string_at(which); - if (s == NULL) { - return (char*)""; - } else { - return unresolved_string_at(which)->as_C_string(); - } + return unresolved_string_at(which)->as_C_string(); } BasicType ConstantPool::basic_type_for_signature_at(int which) { @@ -1828,7 +1823,7 @@ void ConstantPool::patch_resolved_references(GrowableArray* cp_patches) // explicitly, because it may require scavenging. int obj_index = cp_to_object_index(index); pseudo_string_at_put(index, obj_index, patch()); - DEBUG_ONLY(cp_patches->at_put(index, Handle());) + DEBUG_ONLY(cp_patches->at_put(index, Handle());) } } #ifdef ASSERT diff --git a/hotspot/src/share/vm/oops/constantPool.hpp b/hotspot/src/share/vm/oops/constantPool.hpp index 359fa22d522..538c78fe8fb 100644 --- a/hotspot/src/share/vm/oops/constantPool.hpp +++ b/hotspot/src/share/vm/oops/constantPool.hpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -48,17 +48,21 @@ class SymbolHashMap; class CPSlot VALUE_OBJ_CLASS_SPEC { intptr_t _ptr; public: + enum TagBits { _resolved_value = 0, _symbol_bit = 1, _pseudo_bit = 2, _symbol_mask = 3 }; + CPSlot(intptr_t ptr): _ptr(ptr) {} CPSlot(Klass* ptr): _ptr((intptr_t)ptr) {} - CPSlot(Symbol* ptr): _ptr((intptr_t)ptr | 1) {} + CPSlot(Symbol* ptr): _ptr((intptr_t)ptr | _symbol_bit) {} + CPSlot(Symbol* ptr, int tag_bits): _ptr((intptr_t)ptr | tag_bits) {} intptr_t value() { return _ptr; } - bool is_resolved() { return (_ptr & 1) == 0; } - bool is_unresolved() { return (_ptr & 1) == 1; } + bool is_resolved() { return (_ptr & _symbol_bit ) == _resolved_value; } + bool is_unresolved() { return (_ptr & _symbol_bit ) != _resolved_value; } + bool is_pseudo_string() { return (_ptr & _symbol_mask) == _symbol_bit + _pseudo_bit; } Symbol* get_symbol() { assert(is_unresolved(), "bad call"); - return (Symbol*)(_ptr & ~1); + return (Symbol*)(_ptr & ~_symbol_mask); } Klass* get_klass() { assert(is_resolved(), "bad call"); @@ -261,7 +265,7 @@ class ConstantPool : public Metadata { void unresolved_string_at_put(int which, Symbol* s) { release_tag_at_put(which, JVM_CONSTANT_String); - *symbol_at_addr(which) = s; + slot_at_put(which, CPSlot(s, CPSlot::_symbol_bit)); } void int_at_put(int which, jint i) { @@ -405,20 +409,18 @@ class ConstantPool : public Metadata { // use pseudo-strings to link themselves to related metaobjects. bool is_pseudo_string_at(int which) { - // A pseudo string is a string that doesn't have a symbol in the cpSlot - return unresolved_string_at(which) == NULL; + assert(tag_at(which).is_string(), "Corrupted constant pool"); + return slot_at(which).is_pseudo_string(); } oop pseudo_string_at(int which, int obj_index) { - assert(tag_at(which).is_string(), "Corrupted constant pool"); - assert(unresolved_string_at(which) == NULL, "shouldn't have symbol"); + assert(is_pseudo_string_at(which), "must be a pseudo-string"); oop s = resolved_references()->obj_at(obj_index); return s; } oop pseudo_string_at(int which) { - assert(tag_at(which).is_string(), "Corrupted constant pool"); - assert(unresolved_string_at(which) == NULL, "shouldn't have symbol"); + assert(is_pseudo_string_at(which), "must be a pseudo-string"); int obj_index = cp_to_object_index(which); oop s = resolved_references()->obj_at(obj_index); return s; @@ -426,7 +428,8 @@ class ConstantPool : public Metadata { void pseudo_string_at_put(int which, int obj_index, oop x) { assert(tag_at(which).is_string(), "Corrupted constant pool"); - unresolved_string_at_put(which, NULL); // indicates patched string + Symbol* sym = unresolved_string_at(which); + slot_at_put(which, CPSlot(sym, (CPSlot::_symbol_bit | CPSlot::_pseudo_bit))); string_at_put(which, obj_index, x); // this works just fine } @@ -443,15 +446,14 @@ class ConstantPool : public Metadata { Symbol* unresolved_string_at(int which) { assert(tag_at(which).is_string(), "Corrupted constant pool"); - Symbol* s = *symbol_at_addr(which); - return s; + Symbol* sym = slot_at(which).get_symbol(); + return sym; } // Returns an UTF8 for a CONSTANT_String entry at a given index. // UTF8 char* representation was chosen to avoid conversion of // java_lang_Strings at resolved entries into Symbol*s // or vice versa. - // Caller is responsible for checking for pseudo-strings. char* string_at_noresolve(int which); jint name_and_type_at(int which) { diff --git a/hotspot/src/share/vm/prims/methodComparator.cpp b/hotspot/src/share/vm/prims/methodComparator.cpp index c5b73f27207..1477a35746a 100644 --- a/hotspot/src/share/vm/prims/methodComparator.cpp +++ b/hotspot/src/share/vm/prims/methodComparator.cpp @@ -1,5 +1,5 @@ /* - * Copyright (c) 2000, 2012, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2000, 2015, 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 @@ -405,6 +405,8 @@ bool MethodComparator::pool_constants_same(int cpi_old, int cpi_new) { if (strcmp(_old_cp->string_at_noresolve(cpi_old), _new_cp->string_at_noresolve(cpi_new)) != 0) return false; + if (_old_cp->is_pseudo_string_at(cpi_old) || _new_cp->is_pseudo_string_at(cpi_new)) + return (_old_cp->is_pseudo_string_at(cpi_old) == _new_cp->is_pseudo_string_at(cpi_new)); } else if (tag_old.is_klass() || tag_old.is_unresolved_klass()) { // tag_old should be klass - 4881222 if (! (tag_new.is_unresolved_klass() || tag_new.is_klass())) diff --git a/hotspot/test/serviceability/jvmti/TestLambdaFormRetransformation.java b/hotspot/test/serviceability/jvmti/TestLambdaFormRetransformation.java new file mode 100644 index 00000000000..9ba16d810fa --- /dev/null +++ b/hotspot/test/serviceability/jvmti/TestLambdaFormRetransformation.java @@ -0,0 +1,138 @@ + +/* + * Copyright (c) 2015, 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 8008678 + * @summary JSR 292: constant pool reconstitution must support pseudo strings + * @library /testlibrary + * @compile -XDignore.symbol.file TestLambdaFormRetransformation.java + * @run main TestLambdaFormRetransformation + */ + +import java.io.IOException; +import java.lang.instrument.ClassFileTransformer; +import java.lang.instrument.IllegalClassFormatException; +import java.lang.instrument.Instrumentation; +import java.lang.instrument.UnmodifiableClassException; +import java.nio.file.Files; +import java.nio.file.Path; +import java.nio.file.Paths; +import java.security.ProtectionDomain; +import java.util.Arrays; + +import com.oracle.java.testlibrary.ExitCode; +import com.oracle.java.testlibrary.OutputAnalyzer; +import com.oracle.java.testlibrary.ProcessTools; + +public class TestLambdaFormRetransformation { + private static String MANIFEST = String.format("Manifest-Version: 1.0\n" + + "Premain-Class: %s\n" + + "Can-Retransform-Classes: true\n", + Agent.class.getName()); + + private static String CP = System.getProperty("test.classes"); + + public static void main(String args[]) throws Throwable { + Path agent = TestLambdaFormRetransformation.buildAgent(); + OutputAnalyzer oa = ProcessTools.executeTestJvm("-javaagent:" + + agent.toAbsolutePath().toString(), "-version"); + oa.shouldHaveExitValue(ExitCode.OK.value); + } + + private static Path buildAgent() throws IOException { + Path manifest = TestLambdaFormRetransformation.createManifest(); + Path jar = Files.createTempFile(Paths.get("."), null, ".jar"); + + String[] args = new String[] { + "-cfm", + jar.toAbsolutePath().toString(), + manifest.toAbsolutePath().toString(), + "-C", + TestLambdaFormRetransformation.CP, + Agent.class.getName() + ".class" + }; + + sun.tools.jar.Main jarTool = new sun.tools.jar.Main(System.out, System.err, "jar"); + + if (!jarTool.run(args)) { + throw new Error("jar failed: args=" + Arrays.toString(args)); + } + return jar; + } + + private static Path createManifest() throws IOException { + Path manifest = Files.createTempFile(Paths.get("."), null, ".mf"); + byte[] manifestBytes = TestLambdaFormRetransformation.MANIFEST.getBytes(); + Files.write(manifest, manifestBytes); + return manifest; + } +} + +class Agent implements ClassFileTransformer { + private static Runnable lambda = () -> { + System.out.println("I'll crash you!"); + }; + + public static void premain(String args, Instrumentation instrumentation) { + if (!instrumentation.isRetransformClassesSupported()) { + System.out.println("Class retransformation is not supported."); + return; + } + System.out.println("Calling lambda to ensure that lambda forms were created"); + + Agent.lambda.run(); + + System.out.println("Registering class file transformer"); + + instrumentation.addTransformer(new Agent()); + + for (Class c : instrumentation.getAllLoadedClasses()) { + if (c.getName().contains("LambdaForm") && + instrumentation.isModifiableClass(c)) { + System.out.format("We've found a modifiable lambda form: %s%n", c.getName()); + try { + instrumentation.retransformClasses(c); + } catch (UnmodifiableClassException e) { + throw new AssertionError("Modification of modifiable class " + + "caused UnmodifiableClassException", e); + } + } + } + } + + public static void main(String args[]) { + } + + @Override + public byte[] transform(ClassLoader loader, + String className, + Class classBeingRedefined, + ProtectionDomain protectionDomain, + byte[] classfileBuffer + ) throws IllegalClassFormatException { + System.out.println("Transforming " + className); + return classfileBuffer.clone(); + } +} From 73d940848baa4c33929633d0b74f27f844f53c2e Mon Sep 17 00:00:00 2001 From: Attila Szegedi Date: Fri, 30 Jan 2015 15:03:56 +0100 Subject: [PATCH 26/58] 8072000: New compiler warning after JDK-8067139 Reviewed-by: hannesw, sundar --- .../share/classes/jdk/nashorn/internal/codegen/Lower.java | 1 + 1 file changed, 1 insertion(+) diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/codegen/Lower.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/codegen/Lower.java index 8bb947cd346..b7f95aa617c 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/codegen/Lower.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/codegen/Lower.java @@ -326,6 +326,7 @@ final class Lower extends NodeOperatorVisitor implements Lo return addStatement(throwNode); //ThrowNodes are always terminal, marked as such in constructor } + @SuppressWarnings("unchecked") private static T ensureUniqueNamesIn(final T node) { return (T)node.accept(new NodeVisitor(new LexicalContext()) { @Override From c2fe1ca4702f90cdb5f046acf28bfd7e4e9382e9 Mon Sep 17 00:00:00 2001 From: Brian Burkhalter Date: Fri, 30 Jan 2015 10:23:45 -0800 Subject: [PATCH 27/58] 8067669: Documentation for methods in Number incomplete regarding too large values Remove statments about rounding and truncation from the *Valud() methods Reviewed-by: rriggs, darcy, alundblad --- .../share/classes/java/lang/Number.java | 22 +++++++------------ 1 file changed, 8 insertions(+), 14 deletions(-) diff --git a/jdk/src/java.base/share/classes/java/lang/Number.java b/jdk/src/java.base/share/classes/java/lang/Number.java index b89ed7190f0..019ed9c56c3 100644 --- a/jdk/src/java.base/share/classes/java/lang/Number.java +++ b/jdk/src/java.base/share/classes/java/lang/Number.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1994, 2011, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1994, 2015, 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 @@ -37,7 +37,7 @@ package java.lang; * * For platform classes, the conversion is often analogous to a * narrowing primitive conversion or a widening primitive conversion - * as defining in The Java™ Language Specification + * as defined in The Java™ Language Specification * for converting between primitive types. Therefore, conversions may * lose information about the overall magnitude of a numeric value, may * lose precision, and may even return a result of a different sign @@ -54,8 +54,7 @@ package java.lang; */ public abstract class Number implements java.io.Serializable { /** - * Returns the value of the specified number as an {@code int}, - * which may involve rounding or truncation. + * Returns the value of the specified number as an {@code int}. * * @return the numeric value represented by this object after conversion * to type {@code int}. @@ -63,8 +62,7 @@ public abstract class Number implements java.io.Serializable { public abstract int intValue(); /** - * Returns the value of the specified number as a {@code long}, - * which may involve rounding or truncation. + * Returns the value of the specified number as a {@code long}. * * @return the numeric value represented by this object after conversion * to type {@code long}. @@ -72,8 +70,7 @@ public abstract class Number implements java.io.Serializable { public abstract long longValue(); /** - * Returns the value of the specified number as a {@code float}, - * which may involve rounding. + * Returns the value of the specified number as a {@code float}. * * @return the numeric value represented by this object after conversion * to type {@code float}. @@ -81,8 +78,7 @@ public abstract class Number implements java.io.Serializable { public abstract float floatValue(); /** - * Returns the value of the specified number as a {@code double}, - * which may involve rounding. + * Returns the value of the specified number as a {@code double}. * * @return the numeric value represented by this object after conversion * to type {@code double}. @@ -90,8 +86,7 @@ public abstract class Number implements java.io.Serializable { public abstract double doubleValue(); /** - * Returns the value of the specified number as a {@code byte}, - * which may involve rounding or truncation. + * Returns the value of the specified number as a {@code byte}. * *

This implementation returns the result of {@link #intValue} cast * to a {@code byte}. @@ -105,8 +100,7 @@ public abstract class Number implements java.io.Serializable { } /** - * Returns the value of the specified number as a {@code short}, - * which may involve rounding or truncation. + * Returns the value of the specified number as a {@code short}. * *

This implementation returns the result of {@link #intValue} cast * to a {@code short}. From 2a8d5c46e1ccc4e6b9e0e5e5d0d6edc50b6d316b Mon Sep 17 00:00:00 2001 From: Brian Burkhalter Date: Fri, 30 Jan 2015 12:56:12 -0800 Subject: [PATCH 28/58] 6880737: (fs) FileLock constructors don't throw NPE if the channel argument is null Throw IllegalArgumentException if the channel parameter is null Reviewed-by: alanb --- .../classes/java/nio/channels/FileLock.java | 3 + .../FileLock/FileLockConstructor.java | 169 ++++++++++++++++++ 2 files changed, 172 insertions(+) create mode 100644 jdk/test/java/nio/channels/FileLock/FileLockConstructor.java diff --git a/jdk/src/java.base/share/classes/java/nio/channels/FileLock.java b/jdk/src/java.base/share/classes/java/nio/channels/FileLock.java index 156071c99e7..73e351126ca 100644 --- a/jdk/src/java.base/share/classes/java/nio/channels/FileLock.java +++ b/jdk/src/java.base/share/classes/java/nio/channels/FileLock.java @@ -26,6 +26,7 @@ package java.nio.channels; import java.io.IOException; +import java.util.Objects; /** * A token representing a lock on a region of a file. @@ -147,6 +148,7 @@ public abstract class FileLock implements AutoCloseable { protected FileLock(FileChannel channel, long position, long size, boolean shared) { + Objects.requireNonNull(channel, "Null channel"); if (position < 0) throw new IllegalArgumentException("Negative position"); if (size < 0) @@ -185,6 +187,7 @@ public abstract class FileLock implements AutoCloseable { protected FileLock(AsynchronousFileChannel channel, long position, long size, boolean shared) { + Objects.requireNonNull(channel, "Null channel"); if (position < 0) throw new IllegalArgumentException("Negative position"); if (size < 0) diff --git a/jdk/test/java/nio/channels/FileLock/FileLockConstructor.java b/jdk/test/java/nio/channels/FileLock/FileLockConstructor.java new file mode 100644 index 00000000000..4655ebf80e9 --- /dev/null +++ b/jdk/test/java/nio/channels/FileLock/FileLockConstructor.java @@ -0,0 +1,169 @@ +/* + * Copyright (c) 2015, 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. + */ + +import java.io.File; +import java.io.IOException; +import java.nio.channels.AsynchronousFileChannel; +import java.nio.channels.FileChannel; +import java.nio.channels.FileLock; +import java.nio.file.StandardOpenOption; + +/* + * @test + * @bug 6880737 + * @summary Test FileLock constructor parameter validation. + */ +public class FileLockConstructor { + public static void main(String[] args) throws IOException { + FileLock fileLock = null; + int failures = 0; + + // null FileChannel + boolean exceptionThrown = false; + try { + fileLock = new FileLockSub((FileChannel)null, 0, 0, false); + } catch (NullPointerException npe) { + exceptionThrown = true; + } + if (!exceptionThrown) { + System.err.println("FileLock constructor did not throw NPE for null FileChannel"); + failures++; + } + + // null AsynchronousFileChannel + exceptionThrown = false; + try { + fileLock = new FileLockSub((AsynchronousFileChannel)null, 0, 0, true); + } catch (NullPointerException npe) { + exceptionThrown = true; + } + if (!exceptionThrown) { + System.err.println("FileLock constructor did not throw NPE for null AsynchronousFileChannel"); + failures++; + } + + // create temporary file + File tmpFile = File.createTempFile("FileLock", "tmp"); + tmpFile.deleteOnExit(); + + // position and size preconditions + long[][] posAndSize = new long[][] { + {0, 42}, // valid + {-1, 42}, // invalid: position < 0 + {0, -1}, // invalid: size < 0 + {Long.MAX_VALUE, 1} // invalid: position + size < 0 + }; + + // test position and size preconditions for FileChannel case + try (FileChannel syncChannel = FileChannel.open(tmpFile.toPath(), + StandardOpenOption.READ, StandardOpenOption.WRITE)) { + + for (int i = 0; i < posAndSize.length; i++) { + boolean preconditionsHold = i == 0; + exceptionThrown = false; + try { + fileLock = new FileLockSub(syncChannel, posAndSize[i][0], + posAndSize[i][1], true); + } catch (IllegalArgumentException iae) { + exceptionThrown = true; + } catch (Exception e) { + System.err.println("Unexpected exception \"" + e + "\" caught" + + " for position " + posAndSize[i][0] + " and size " + + posAndSize[i][1] + " for FileChannel variant"); + failures++; + continue; + } + if (preconditionsHold && exceptionThrown) { + System.err.println("FileLock constructor incorrectly threw IAE" + + " for position " + posAndSize[i][0] + " and size " + + posAndSize[i][1] + " for FileChannel variant"); + failures++; + } else if (!preconditionsHold && !exceptionThrown) { + System.err.println("FileLock constructor did not throw IAE" + + " for position " + posAndSize[i][0] + " and size " + + posAndSize[i][1] + " for FileChannel variant"); + failures++; + } + } + } + + // test position and size preconditions for AsynchronousFileChannel case + try (AsynchronousFileChannel asyncChannel + = AsynchronousFileChannel.open(tmpFile.toPath(), + StandardOpenOption.READ, StandardOpenOption.WRITE)) { + for (int i = 0; i < posAndSize.length; i++) { + boolean preconditionsHold = i == 0; + exceptionThrown = false; + try { + fileLock = new FileLockSub(asyncChannel, posAndSize[i][0], + posAndSize[i][1], true); + } catch (IllegalArgumentException iae) { + exceptionThrown = true; + } catch (Exception e) { + System.err.println("Unexpected exception \"" + e + "\" caught" + + " for position " + posAndSize[i][0] + " and size " + + posAndSize[i][1] + " for AsynchronousFileChannel variant"); + failures++; + continue; + } + if (preconditionsHold && exceptionThrown) { + System.err.println("FileLock constructor incorrectly threw IAE" + + " for position " + posAndSize[i][0] + " and size " + + posAndSize[i][1] + " for AsynchronousFileChannel variant"); + failures++; + } else if (!preconditionsHold && !exceptionThrown) { + System.err.println("FileLock constructor did not throw IAE" + + " for position " + posAndSize[i][0] + " and size " + + posAndSize[i][1] + " for AsynchronousFileChannel variant"); + failures++; + } + } + } + + if (failures > 0) { + throw new RuntimeException("Incurred " + failures + + " failures while testing FileLock."); + } + } +} + +class FileLockSub extends FileLock { + FileLockSub(FileChannel channel, long position, long size, boolean shared) { + super(channel, position, size, shared); + } + + FileLockSub(AsynchronousFileChannel channel, long position, long size, + boolean shared) { + super(channel, position, size, shared); + } + + @Override + public boolean isValid() { + return false; + } + + @Override + public void release() throws IOException { + // do nothing + } +} From 0854df7b86f3ab2b5fa3c3550c857c90a4bfd96c Mon Sep 17 00:00:00 2001 From: Roger Riggs Date: Fri, 30 Jan 2015 16:13:04 -0500 Subject: [PATCH 29/58] 8055330: (process spec) ProcessBuilder.start and Runtime.exec should throw UnsupportedOperationException on platforms that don't support Clarify optional behavior and the exception thrown when not supported Reviewed-by: dfuchs, martin --- .../java.base/share/classes/java/lang/ProcessBuilder.java | 6 ++++++ jdk/src/java.base/share/classes/java/lang/Runtime.java | 6 ++++++ 2 files changed, 12 insertions(+) diff --git a/jdk/src/java.base/share/classes/java/lang/ProcessBuilder.java b/jdk/src/java.base/share/classes/java/lang/ProcessBuilder.java index 8d4576bd40c..a62594383fc 100644 --- a/jdk/src/java.base/share/classes/java/lang/ProcessBuilder.java +++ b/jdk/src/java.base/share/classes/java/lang/ProcessBuilder.java @@ -951,6 +951,9 @@ public final class ProcessBuilder * {@code command} array as its argument. This may result in * a {@link SecurityException} being thrown. * + *

If the operating system does not support the creation of + * processes, an {@link UnsupportedOperationException} will be thrown. + * *

Starting an operating system process is highly system-dependent. * Among the many things that can go wrong are: *

    @@ -998,6 +1001,9 @@ public final class ProcessBuilder * *
* + * @throws UnsupportedOperationException + * If the operating system does not support the creation of processes. + * * @throws IOException if an I/O error occurs * * @see Runtime#exec(String[], String[], java.io.File) diff --git a/jdk/src/java.base/share/classes/java/lang/Runtime.java b/jdk/src/java.base/share/classes/java/lang/Runtime.java index 66c78537d39..98bfe2887b5 100644 --- a/jdk/src/java.base/share/classes/java/lang/Runtime.java +++ b/jdk/src/java.base/share/classes/java/lang/Runtime.java @@ -564,6 +564,9 @@ public class Runtime { * cmdarray as its argument. This may result in a * {@link SecurityException} being thrown. * + *

If the operating system does not support the creation of + * processes, an {@link UnsupportedOperationException} will be thrown. + * *

Starting an operating system process is highly system-dependent. * Among the many things that can go wrong are: *

    @@ -597,6 +600,9 @@ public class Runtime { * {@link SecurityManager#checkExec checkExec} * method doesn't allow creation of the subprocess * + * @throws UnsupportedOperationException + * If the operating system does not support the creation of processes. + * * @throws IOException * If an I/O error occurs * From 24a3c49404fdc8477a082abb9556d97ec8973b45 Mon Sep 17 00:00:00 2001 From: Roger Riggs Date: Fri, 30 Jan 2015 16:13:57 -0500 Subject: [PATCH 30/58] 8068284: Missing @throws in DateTimeFormatterBuilder.appendOffset 8068285: Missing @throws in DateTimeFormatterBuilder.appendInstant 8062803: 'principal' should be 'principle' in java.time package description 8062796: java.time.format.DateTimeFormatter error in API doc example Reviewed-by: lancea, mchung --- .../share/classes/java/time/format/DateTimeFormatter.java | 6 ++++-- .../classes/java/time/format/DateTimeFormatterBuilder.java | 2 ++ jdk/src/java.base/share/classes/java/time/package-info.java | 6 +++--- 3 files changed, 9 insertions(+), 5 deletions(-) diff --git a/jdk/src/java.base/share/classes/java/time/format/DateTimeFormatter.java b/jdk/src/java.base/share/classes/java/time/format/DateTimeFormatter.java index 02994ad5042..4a5a53cd83c 100644 --- a/jdk/src/java.base/share/classes/java/time/format/DateTimeFormatter.java +++ b/jdk/src/java.base/share/classes/java/time/format/DateTimeFormatter.java @@ -117,8 +117,9 @@ import java.util.Set; * {@code parse(CharSequence text, DateTimeFormatter formatter)}. *

    For example: *

    + *  LocalDate date = LocalDate.now();
      *  String text = date.format(formatter);
    - *  LocalDate date = LocalDate.parse(text, formatter);
    + *  LocalDate parsedDate = LocalDate.parse(text, formatter);
      * 
    *

    * In addition to the format, formatters can be created with desired Locale, @@ -265,9 +266,10 @@ import java.util.Set; *

    * For example: *

    + *  LocalDate date = LocalDate.now();
      *  DateTimeFormatter formatter = DateTimeFormatter.ofPattern("yyyy MM dd");
      *  String text = date.format(formatter);
    - *  LocalDate date = LocalDate.parse(text, formatter);
    + *  LocalDate parsedDate = LocalDate.parse(text, formatter);
      * 
    *

    * All letters 'A' to 'Z' and 'a' to 'z' are reserved as pattern letters. The diff --git a/jdk/src/java.base/share/classes/java/time/format/DateTimeFormatterBuilder.java b/jdk/src/java.base/share/classes/java/time/format/DateTimeFormatterBuilder.java index 4616017c8e1..a8f70915012 100644 --- a/jdk/src/java.base/share/classes/java/time/format/DateTimeFormatterBuilder.java +++ b/jdk/src/java.base/share/classes/java/time/format/DateTimeFormatterBuilder.java @@ -849,6 +849,7 @@ public final class DateTimeFormatterBuilder { * @param fractionalDigits the number of fractional second digits to format with, * from 0 to 9, or -1 to use as many digits as necessary * @return this, for chaining, not null + * @throws IllegalArgumentException if the number of fractional digits is invalid */ public DateTimeFormatterBuilder appendInstant(int fractionalDigits) { if (fractionalDigits < -1 || fractionalDigits > 9) { @@ -909,6 +910,7 @@ public final class DateTimeFormatterBuilder { * @param pattern the pattern to use, not null * @param noOffsetText the text to use when the offset is zero, not null * @return this, for chaining, not null + * @throws IllegalArgumentException if the pattern is invalid */ public DateTimeFormatterBuilder appendOffset(String pattern, String noOffsetText) { appendInternal(new OffsetIdPrinterParser(pattern, noOffsetText)); diff --git a/jdk/src/java.base/share/classes/java/time/package-info.java b/jdk/src/java.base/share/classes/java/time/package-info.java index b97a56b9c36..cea222126f1 100644 --- a/jdk/src/java.base/share/classes/java/time/package-info.java +++ b/jdk/src/java.base/share/classes/java/time/package-info.java @@ -65,7 +65,7 @@ * The main API for dates, times, instants, and durations. *

    *

    - * The classes defined here represent the principal date-time concepts, + * The classes defined here represent the principle date-time concepts, * including instants, durations, dates, times, time-zones and periods. * They are based on the ISO calendar system, which is the de facto world * calendar following the proleptic Gregorian rules. @@ -247,8 +247,8 @@ *

*

* Multiple calendar systems is an awkward addition to the design challenges. - * The first principal is that most users want the standard ISO calendar system. - * As such, the main classes are ISO-only. The second principal is that most of those that want a + * The first principle is that most users want the standard ISO calendar system. + * As such, the main classes are ISO-only. The second principle is that most of those that want a * non-ISO calendar system want it for user interaction, thus it is a UI localization issue. * As such, date and time objects should be held as ISO objects in the data model and persistent * storage, only being converted to and from a local calendar for display. From ffe5a2e589f6f97c2d224770991dcd0c2ee634d1 Mon Sep 17 00:00:00 2001 From: Frank Yuan Date: Fri, 30 Jan 2015 17:07:53 -0800 Subject: [PATCH 31/58] 8051709: Convert JAXP function tests: javax.xml.datatype to jtreg (testng) tests Reviewed-by: lancea --- .../xml/datatype/ptests/DurationTest.java | 585 ++++++++++++++++++ .../ptests/XMLGregorianCalendarTest.java | 356 +++++++++++ 2 files changed, 941 insertions(+) create mode 100644 jaxp/test/javax/xml/jaxp/functional/javax/xml/datatype/ptests/DurationTest.java create mode 100644 jaxp/test/javax/xml/jaxp/functional/javax/xml/datatype/ptests/XMLGregorianCalendarTest.java diff --git a/jaxp/test/javax/xml/jaxp/functional/javax/xml/datatype/ptests/DurationTest.java b/jaxp/test/javax/xml/jaxp/functional/javax/xml/datatype/ptests/DurationTest.java new file mode 100644 index 00000000000..f18cf9c6c29 --- /dev/null +++ b/jaxp/test/javax/xml/jaxp/functional/javax/xml/datatype/ptests/DurationTest.java @@ -0,0 +1,585 @@ +/* + * Copyright (c) 1999, 2015, 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. + */ + +package javax.xml.datatype.ptests; + +import static javax.xml.datatype.DatatypeConstants.DAYS; +import static javax.xml.datatype.DatatypeConstants.HOURS; +import static javax.xml.datatype.DatatypeConstants.MINUTES; +import static javax.xml.datatype.DatatypeConstants.MONTHS; +import static javax.xml.datatype.DatatypeConstants.SECONDS; +import static javax.xml.datatype.DatatypeConstants.YEARS; +import static org.testng.Assert.assertEquals; +import static org.testng.Assert.assertFalse; +import static org.testng.Assert.assertTrue; + +import java.math.BigDecimal; +import java.math.BigInteger; +import java.util.Calendar; +import java.util.function.Function; + +import javax.xml.datatype.DatatypeConfigurationException; +import javax.xml.datatype.DatatypeConstants; +import javax.xml.datatype.DatatypeFactory; +import javax.xml.datatype.Duration; +import javax.xml.namespace.QName; + +import jaxp.library.JAXPBaseTest; + +import org.testng.Assert; +import org.testng.annotations.BeforeClass; +import org.testng.annotations.DataProvider; +import org.testng.annotations.Test; + +/* + * @summary Class containing the test cases for Duration. + */ +public class DurationTest extends JAXPBaseTest { + + private DatatypeFactory datatypeFactory; + + /* + * Setup. + */ + @BeforeClass + public void setup() throws DatatypeConfigurationException { + datatypeFactory = DatatypeFactory.newInstance(); + } + + @DataProvider(name = "legal-number-duration") + public Object[][] getLegalNumberDuration() { + return new Object[][] { + // is positive, year, month, day, hour, minute, second + { true, 1, 1, 1, 1, 1, 1 }, + { false, 1, 1, 1, 1, 1, 1 }, + { true, 1, 0, 0, 0, 0, 0 }, + { false, 1, 0, 0, 0, 0, 0 } + }; + } + + /* + * Test for constructor Duration(boolean isPositive,int years,int months, + * int days,int hours,int minutes,int seconds). + */ + @Test(dataProvider = "legal-number-duration") + public void checkNumberDurationPos(boolean isPositive, int years, int months, int days, int hours, int minutes, int seconds) { + datatypeFactory.newDuration(isPositive, years, months, days, hours, minutes, seconds); + } + + @DataProvider(name = "illegal-number-duration") + public Object[][] getIllegalNumberDuration() { + return new Object[][] { + // is positive, year, month, day, hour, minute, second + { true, 1, 1, -1, 1, 1, 1 }, + { false, 1, 1, -1, 1, 1, 1 }, + { true, undef, undef, undef, undef, undef, undef }, + { false, undef, undef, undef, undef, undef, undef } + }; + } + + /* + * Test for constructor Duration(boolean isPositive,int years,int months, + * int days,int hours,int minutes,int seconds), if any of the fields is + * negative should throw IllegalArgumentException. + */ + @Test(expectedExceptions = IllegalArgumentException.class, dataProvider = "illegal-number-duration") + public void checkDurationNumberNeg(boolean isPositive, int years, int months, int days, int hours, int minutes, int seconds) { + datatypeFactory.newDuration(isPositive, years, months, days, hours, minutes, seconds); + } + + @DataProvider(name = "legal-bigint-duration") + public Object[][] getLegalBigIntegerDuration() { + return new Object[][] { + // is positive, year, month, day, hour, minute, second + { true, zero, zero, zero, zero, zero, new BigDecimal(zero) }, + { false, zero, zero, zero, zero, zero, new BigDecimal(zero) }, + { true, one, one, one, one, one, new BigDecimal(one) }, + { false, one, one, one, one, one, new BigDecimal(one) }, + { true, null, null, null, null, null, new BigDecimal(one) }, + { false, null, null, null, null, null, new BigDecimal(one) } }; + } + + /* + * Test for constructor Duration(boolean isPositive,BigInteger + * years,BigInteger months, BigInteger days,BigInteger hours,BigInteger + * minutes,BigDecimal seconds). + */ + @Test(dataProvider = "legal-bigint-duration") + public void checkBigIntegerDurationPos(boolean isPositive, BigInteger years, BigInteger months, BigInteger days, BigInteger hours, BigInteger minutes, + BigDecimal seconds) { + datatypeFactory.newDuration(isPositive, years, months, days, hours, minutes, seconds); + } + + @DataProvider(name = "illegal-bigint-duration") + public Object[][] getIllegalBigIntegerDuration() { + return new Object[][] { + // is positive, year, month, day, hour, minute, second + { true, null, null, null, null, null, null }, + { false, null, null, null, null, null, null } + }; + } + + /* + * Test for constructor Duration(boolean isPositive,BigInteger + * years,BigInteger months, BigInteger days,BigInteger hours,BigInteger + * minutes,BigDecimal seconds), if all the fields are null should throw + * IllegalArgumentException. + */ + @Test(expectedExceptions = IllegalArgumentException.class, dataProvider = "illegal-bigint-duration") + public void checkBigIntegerDurationNeg(boolean isPositive, BigInteger years, BigInteger months, BigInteger days, BigInteger hours, BigInteger minutes, + BigDecimal seconds) { + datatypeFactory.newDuration(isPositive, years, months, days, hours, minutes, seconds); + } + + @DataProvider(name = "legal-millisec-duration") + public Object[][] getLegalMilliSecondDuration() { + return new Object[][] { { 1000000 }, { 0 }, { Long.MAX_VALUE }, { Long.MIN_VALUE } + + }; + } + + /* + * Test for constructor Duration(long durationInMilliSeconds) + */ + @Test(dataProvider = "legal-millisec-duration") + public void checkMilliSecondDuration(long millisec) { + datatypeFactory.newDuration(millisec); + } + + @DataProvider(name = "legal-lexical-duration") + public Object[][] getLegalLexicalDuration() { + return new Object[][] { { "P1Y1M1DT1H1M1S" }, { "-P1Y1M1DT1H1M1S" } }; + } + + /* + * Test for constructor Duration(java.lang.String lexicalRepresentation) + */ + @Test(dataProvider = "legal-lexical-duration") + public void checkLexicalDurationPos(String lexRepresentation) { + datatypeFactory.newDuration(lexRepresentation); + } + + @DataProvider(name = "illegal-lexical-duration") + public Object[][] getIllegalLexicalDuration() { + return new Object[][] { + { null }, + { "P1Y1M1DT1H1M1S " }, + { " P1Y1M1DT1H1M1S" }, + { "X1Y1M1DT1H1M1S" }, + { "" }, + { "P1Y2MT" } // The designator 'T' shall be absent if all of the time items are absent in "PnYnMnDTnHnMnS" + }; + } + + /* + * Test for constructor Duration(java.lang.String lexicalRepresentation), + * null should throw NullPointerException, invalid lex should throw + * IllegalArgumentException + */ + @Test(expectedExceptions = { NullPointerException.class, IllegalArgumentException.class }, dataProvider = "illegal-lexical-duration") + public void checkLexicalDurationNeg(String lexRepresentation) { + datatypeFactory.newDuration(lexRepresentation); + } + + @DataProvider(name = "equal-duration") + public Object[][] getEqualDurations() { + return new Object[][] { { "P1Y1M1DT1H1M1S", "P1Y1M1DT1H1M1S" } }; + } + + /* + * Test for compare() both durations valid and equal. + */ + @Test(dataProvider = "equal-duration") + public void checkDurationEqual(String lexRepresentation1, String lexRepresentation2) { + Duration duration1 = datatypeFactory.newDuration(lexRepresentation1); + Duration duration2 = datatypeFactory.newDuration(lexRepresentation2); + assertTrue(duration1.equals(duration2)); + } + + @DataProvider(name = "greater-duration") + public Object[][] getGreaterDuration() { + return new Object[][] { + { "P1Y1M1DT1H1M2S", "P1Y1M1DT1H1M1S" }, + { "P1Y1M1DT1H1M1S", "-P1Y1M1DT1H1M2S" }, + { "P1Y1M1DT1H1M2S", "-P1Y1M1DT1H1M1S" }, + { "-P1Y1M1DT1H1M1S", "-P1Y1M1DT1H1M2S" }, }; + } + + /* + * Test for compare() both durations valid and lhs > rhs. + */ + @Test(dataProvider = "greater-duration") + public void checkDurationCompare(String lexRepresentation1, String lexRepresentation2) { + Duration duration1 = datatypeFactory.newDuration(lexRepresentation1); + Duration duration2 = datatypeFactory.newDuration(lexRepresentation2); + assertTrue(duration1.compare(duration2) == DatatypeConstants.GREATER); + } + + @DataProvider(name = "not-equal-duration") + public Object[][] getNotEqualDurations() { + return new Object[][] { + { "P1Y1M1DT1H1M1S", "-P1Y1M1DT1H1M1S" }, + { "P2Y1M1DT1H1M1S", "P1Y1M1DT1H1M1S" } }; + } + + /* + * Test for equals() both durations valid and lhs not equals rhs. + */ + @Test(dataProvider = "not-equal-duration") + public void checkDurationNotEqual(String lexRepresentation1, String lexRepresentation2) { + Duration duration1 = datatypeFactory.newDuration(lexRepresentation1); + Duration duration2 = datatypeFactory.newDuration(lexRepresentation2); + Assert.assertNotEquals(duration1, duration2); + } + + @DataProvider(name = "duration-sign") + public Object[][] getDurationAndSign() { + return new Object[][] { + { "P0Y0M0DT0H0M0S", 0 }, + { "P1Y0M0DT0H0M0S", 1 }, + { "-P1Y0M0DT0H0M0S", -1 } }; + } + + /* + * Test for Duration.getSign(). + */ + @Test(dataProvider = "duration-sign") + public void checkDurationSign(String lexRepresentation, int sign) { + Duration duration = datatypeFactory.newDuration(lexRepresentation); + assertEquals(duration.getSign(), sign); + } + + /* + * Test for Duration.negate(). + */ + @Test + public void checkDurationNegate() { + Duration durationPos = datatypeFactory.newDuration("P1Y0M0DT0H0M0S"); + Duration durationNeg = datatypeFactory.newDuration("-P1Y0M0DT0H0M0S"); + + assertEquals(durationPos.negate(), durationNeg); + assertEquals(durationNeg.negate(), durationPos); + assertEquals(durationPos.negate().negate(), durationPos); + + } + + /* + * Test for Duration.isShorterThan(Duration) and + * Duration.isLongerThan(Duration). + */ + @Test + public void checkDurationShorterLonger() { + Duration shorter = datatypeFactory.newDuration("P1Y1M1DT1H1M1S"); + Duration longer = datatypeFactory.newDuration("P2Y1M1DT1H1M1S"); + + assertTrue(shorter.isShorterThan(longer)); + assertFalse(longer.isShorterThan(shorter)); + assertFalse(shorter.isShorterThan(shorter)); + + assertTrue(longer.isLongerThan(shorter)); + assertFalse(shorter.isLongerThan(longer)); + assertFalse(shorter.isLongerThan(shorter)); + } + + /* + * Test for Duration.isSet(). + */ + @Test + public void checkDurationIsSet() { + Duration duration1 = datatypeFactory.newDuration(true, 1, 1, 1, 1, 1, 1); + Duration duration2 = datatypeFactory.newDuration(true, 0, 0, 0, 0, 0, 0); + + assertTrue(duration1.isSet(YEARS)); + assertTrue(duration1.isSet(MONTHS)); + assertTrue(duration1.isSet(DAYS)); + assertTrue(duration1.isSet(HOURS)); + assertTrue(duration1.isSet(MINUTES)); + assertTrue(duration1.isSet(SECONDS)); + + assertTrue(duration2.isSet(YEARS)); + assertTrue(duration2.isSet(MONTHS)); + assertTrue(duration2.isSet(DAYS)); + assertTrue(duration2.isSet(HOURS)); + assertTrue(duration2.isSet(MINUTES)); + assertTrue(duration2.isSet(SECONDS)); + + Duration duration66 = datatypeFactory.newDuration(true, null, null, zero, null, null, null); + assertFalse(duration66.isSet(YEARS)); + assertFalse(duration66.isSet(MONTHS)); + assertFalse(duration66.isSet(HOURS)); + assertFalse(duration66.isSet(MINUTES)); + assertFalse(duration66.isSet(SECONDS)); + + Duration duration3 = datatypeFactory.newDuration("P1D"); + assertFalse(duration3.isSet(YEARS)); + assertFalse(duration3.isSet(MONTHS)); + assertFalse(duration3.isSet(HOURS)); + assertFalse(duration3.isSet(MINUTES)); + assertFalse(duration3.isSet(SECONDS)); + } + + /* + * Test Duration.isSet(Field) throws NPE if the field parameter is null. + */ + @Test(expectedExceptions = NullPointerException.class) + public void checkDurationIsSetNeg() { + Duration duration = datatypeFactory.newDuration(true, 0, 0, 0, 0, 0, 0); + duration.isSet(null); + } + + /* + * Test for -getField(DatatypeConstants.Field) DatatypeConstants.Field is + * null - throws NPE. + */ + @Test(expectedExceptions = NullPointerException.class) + public void checkDurationGetFieldNeg() { + Duration duration67 = datatypeFactory.newDuration("P1Y1M1DT1H1M1S"); + duration67.getField(null); + } + + @DataProvider(name = "duration-fields") + public Object[][] getDurationAndFields() { + return new Object[][] { + { "P1Y1M1DT1H1M1S", one, one, one, one, one, new BigDecimal(one) }, + { "PT1M", null, null, null, null, one, null }, + { "P1M", null, one, null, null, null, null } }; + } + + /* + * Test for Duration.getField(DatatypeConstants.Field). + */ + @Test(dataProvider = "duration-fields") + public void checkDurationGetField(String lexRepresentation, BigInteger years, BigInteger months, BigInteger days, BigInteger hours, BigInteger minutes, + BigDecimal seconds) { + Duration duration = datatypeFactory.newDuration(lexRepresentation); + + assertEquals(duration.getField(YEARS), years); + assertEquals(duration.getField(MONTHS), months); + assertEquals(duration.getField(DAYS), days); + assertEquals(duration.getField(HOURS), hours); + assertEquals(duration.getField(MINUTES), minutes); + assertEquals(duration.getField(SECONDS), seconds); + } + + @DataProvider(name = "number-string") + public Object[][] getNumberAndString() { + return new Object[][] { + // is positive, year, month, day, hour, minute, second, lexical + { true, 1, 1, 1, 1, 1, 1, "P1Y1M1DT1H1M1S" }, + { false, 1, 1, 1, 1, 1, 1, "-P1Y1M1DT1H1M1S" }, + { true, 0, 0, 0, 0, 0, 0, "P0Y0M0DT0H0M0S" }, + { false, 0, 0, 0, 0, 0, 0, "P0Y0M0DT0H0M0S" } + }; + } + + /* + * Test for - toString(). + */ + @Test(dataProvider = "number-string") + public void checkDurationToString(boolean isPositive, int years, int months, int days, int hours, int minutes, int seconds, String lexical) { + Duration duration = datatypeFactory.newDuration(isPositive, years, months, days, hours, minutes, seconds); + assertEquals(duration.toString(), lexical); + + assertEquals(datatypeFactory.newDuration(duration.toString()), duration); + } + + @DataProvider(name = "duration-field") + public Object[][] getDurationAndField() { + Function getyears = duration -> duration.getYears(); + Function getmonths = duration -> duration.getMonths(); + Function getdays = duration -> duration.getDays(); + Function gethours = duration -> duration.getHours(); + Function getminutes = duration -> duration.getMinutes(); + Function getseconds = duration -> duration.getSeconds(); + return new Object[][] { + { "P1Y1M1DT1H1M1S", getyears, 1 }, + { "P1M1DT1H1M1S", getyears, 0 }, + { "P1Y1M1DT1H1M1S", getmonths, 1 }, + { "P1Y1DT1H1M1S", getmonths, 0 }, + { "P1Y1M1DT1H1M1S", getdays, 1 }, + { "P1Y1MT1H1M1S", getdays, 0 }, + { "P1Y1M1DT1H1M1S", gethours, 1 }, + { "P1Y1M1DT1M1S", gethours, 0 }, + { "P1Y1M1DT1H1M1S", getminutes, 1 }, + { "P1Y1M1DT1H1S", getminutes, 0 }, + { "P1Y1M1DT1H1M1S", getseconds, 1 }, + { "P1Y1M1DT1H1M", getseconds, 0 }, + { "P1Y1M1DT1H1M100000000S", getseconds, 100000000 }, }; + } + + /* + * Test for Duration.getYears(), getMonths(), etc. + */ + @Test(dataProvider = "duration-field") + public void checkDurationGetOneField(String lexRepresentation, Function getter, int value) { + Duration duration = datatypeFactory.newDuration(lexRepresentation); + assertEquals(getter.apply(duration).intValue(), value); + } + + /* + * Test for - getField(SECONDS) + */ + @Test + public void checkDurationGetSecondsField() { + Duration duration85 = datatypeFactory.newDuration("P1Y1M1DT1H1M100000000S"); + assertEquals((duration85.getField(SECONDS)).intValue(), 100000000); + } + + /* + * getTimeInMillis(java.util.Calendar startInstant) returns milliseconds + * between startInstant and startInstant plus this Duration. + */ + @Test + public void checkDurationGetTimeInMillis() { + Duration duration86 = datatypeFactory.newDuration("PT1M1S"); + Calendar calendar86 = Calendar.getInstance(); + assertEquals(duration86.getTimeInMillis(calendar86), 61000); + } + + /* + * getTimeInMillis(java.util.Calendar startInstant) returns milliseconds + * between startInstant and startInstant plus this Duration throws NPE if + * startInstant parameter is null. + */ + @Test(expectedExceptions = NullPointerException.class) + public void checkDurationGetTimeInMillisNeg() { + Duration duration87 = datatypeFactory.newDuration("PT1M1S"); + Calendar calendar87 = null; + duration87.getTimeInMillis(calendar87); + } + + @DataProvider(name = "duration-for-hash") + public Object[][] getDurationsForHash() { + return new Object[][] { + { "P1Y1M1DT1H1M1S", "P1Y1M1DT1H1M1S" }, + { "P1D", "PT24H" }, + { "PT1H", "PT60M" }, + { "PT1M", "PT60S" }, + { "P1Y", "P12M" } }; + } + + /* + * Test for Duration.hashcode(). hashcode() should return same value for + * some equal durations. + */ + @Test(dataProvider = "duration-for-hash") + public void checkDurationHashCode(String lexRepresentation1, String lexRepresentation2) { + Duration duration1 = datatypeFactory.newDuration(lexRepresentation1); + Duration duration2 = datatypeFactory.newDuration(lexRepresentation2); + int hash1 = duration1.hashCode(); + int hash2 = duration2.hashCode(); + assertTrue(hash1 == hash2, " generated hash1 : " + hash1 + " generated hash2 : " + hash2); + } + + @DataProvider(name = "duration-for-add") + public Object[][] getDurationsForAdd() { + return new Object[][] { + // initVal, addVal, resultVal + { "P1Y1M1DT1H1M1S", "P1Y1M1DT1H1M1S", "P2Y2M2DT2H2M2S" }, + { "P1Y1M1DT1H1M1S", "-P1Y1M1DT1H1M1S", "P0Y0M0DT0H0M0S" }, + { "-P1Y1M1DT1H1M1S", "-P1Y1M1DT1H1M1S", "-P2Y2M2DT2H2M2S" }, }; + } + + /* + * Test for add(Duration rhs). + */ + @Test(dataProvider = "duration-for-add") + public void checkDurationAdd(String initVal, String addVal, String result) { + Duration durationInit = datatypeFactory.newDuration(initVal); + Duration durationAdd = datatypeFactory.newDuration(addVal); + Duration durationResult = datatypeFactory.newDuration(result); + + assertEquals(durationInit.add(durationAdd), durationResult); + } + + @DataProvider(name = "duration-for-addneg") + public Object[][] getDurationsForAddNeg() { + return new Object[][] { + // initVal, addVal + { "P1Y1M1DT1H1M1S", null }, + { "P1Y", "-P1D" }, + { "-P1Y", "P1D" }, }; + } + + /* + * Test for add(Duration rhs) 'rhs' is null , should throw NPE. "1 year" + + * "-1 day" or "-1 year" + "1 day" should throw IllegalStateException + */ + @Test(expectedExceptions = { NullPointerException.class, IllegalStateException.class }, dataProvider = "duration-for-addneg") + public void checkDurationAddNeg(String initVal, String addVal) { + Duration durationInit = datatypeFactory.newDuration(initVal); + Duration durationAdd = addVal == null ? null : datatypeFactory.newDuration(addVal); + + durationInit.add(durationAdd); + } + + /* + * Test Duration#compare(Duration duration) with large durations. + * + * Bug # 4972785 UnsupportedOperationException is expected + * + */ + @Test(expectedExceptions = UnsupportedOperationException.class) + public void checkDurationCompareLarge() { + String duration1Lex = "P100000000000000000000D"; + String duration2Lex = "PT2400000000000000000000H"; + + Duration duration1 = datatypeFactory.newDuration(duration1Lex); + Duration duration2 = datatypeFactory.newDuration(duration2Lex); + duration1.compare(duration2); + + } + + /* + * Test Duration#getXMLSchemaType(). + * + * Bug # 5049544 Duration.getXMLSchemaType shall return the correct result + * + */ + @Test + public void checkDurationGetXMLSchemaType() { + // DURATION + Duration duration = datatypeFactory.newDuration("P1Y1M1DT1H1M1S"); + QName duration_xmlSchemaType = duration.getXMLSchemaType(); + assertEquals(duration_xmlSchemaType, DatatypeConstants.DURATION, "Expected DatatypeConstants.DURATION, returned " + duration_xmlSchemaType.toString()); + + // DURATION_DAYTIME + Duration duration_dayTime = datatypeFactory.newDuration("P1DT1H1M1S"); + QName duration_dayTime_xmlSchemaType = duration_dayTime.getXMLSchemaType(); + assertEquals(duration_dayTime_xmlSchemaType, DatatypeConstants.DURATION_DAYTIME, "Expected DatatypeConstants.DURATION_DAYTIME, returned " + + duration_dayTime_xmlSchemaType.toString()); + + // DURATION_YEARMONTH + Duration duration_yearMonth = datatypeFactory.newDuration("P1Y1M"); + QName duration_yearMonth_xmlSchemaType = duration_yearMonth.getXMLSchemaType(); + assertEquals(duration_yearMonth_xmlSchemaType, DatatypeConstants.DURATION_YEARMONTH, "Expected DatatypeConstants.DURATION_YEARMONTH, returned " + + duration_yearMonth_xmlSchemaType.toString()); + + } + + + private final int undef = DatatypeConstants.FIELD_UNDEFINED; + private final BigInteger zero = BigInteger.ZERO; + private final BigInteger one = BigInteger.ONE; + +} diff --git a/jaxp/test/javax/xml/jaxp/functional/javax/xml/datatype/ptests/XMLGregorianCalendarTest.java b/jaxp/test/javax/xml/jaxp/functional/javax/xml/datatype/ptests/XMLGregorianCalendarTest.java new file mode 100644 index 00000000000..3b47d519368 --- /dev/null +++ b/jaxp/test/javax/xml/jaxp/functional/javax/xml/datatype/ptests/XMLGregorianCalendarTest.java @@ -0,0 +1,356 @@ +/* + * Copyright (c) 1999, 2015, 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. + */ + +package javax.xml.datatype.ptests; + +import static java.util.Calendar.HOUR; +import static java.util.Calendar.MINUTE; +import static java.util.Calendar.YEAR; +import static org.testng.Assert.assertEquals; +import static org.testng.Assert.assertTrue; + +import java.util.GregorianCalendar; +import java.util.Locale; +import java.util.TimeZone; + +import javax.xml.datatype.DatatypeConfigurationException; +import javax.xml.datatype.DatatypeConstants; +import javax.xml.datatype.DatatypeFactory; +import javax.xml.datatype.Duration; +import javax.xml.datatype.XMLGregorianCalendar; + +import jaxp.library.JAXPBaseTest; + +import org.testng.Assert; +import org.testng.annotations.BeforeClass; +import org.testng.annotations.DataProvider; +import org.testng.annotations.Test; + +/* + * @bug 5049592 5041845 5048932 5064587 5040542 5049531 5049528 + * @summary Class containing the test cases for XMLGregorianCalendar + */ +public class XMLGregorianCalendarTest extends JAXPBaseTest { + + private DatatypeFactory datatypeFactory; + + @BeforeClass + public void setup() throws DatatypeConfigurationException { + datatypeFactory = DatatypeFactory.newInstance(); + } + + @DataProvider(name = "valid-milliseconds") + public Object[][] getValidMilliSeconds() { + return new Object[][] { { 0 }, { 1 }, { 2 }, { 16 }, { 1000 } }; + } + + /* + * Test DatatypeFactory.newXMLGregorianCalendar(..) with milliseconds > 1. + * + * Bug # 5049592 + * + */ + @Test(dataProvider = "valid-milliseconds") + public void checkNewCalendar(int ms) { + // valid milliseconds + XMLGregorianCalendar calendar = datatypeFactory.newXMLGregorianCalendar(2004, // year + 6, // month + 2, // day + 19, // hour + 20, // minute + 59, // second + ms, // milliseconds + 840 // timezone + ); + // expected success + + assertEquals(calendar.getMillisecond(), ms); + } + + /* + * Test DatatypeFactory.newXMLGregorianCalendarTime(..). + * + * Bug # 5049592 + */ + @Test(dataProvider = "valid-milliseconds") + public void checkNewTime(int ms) { + // valid milliseconds + XMLGregorianCalendar calendar2 = datatypeFactory.newXMLGregorianCalendarTime(19, // hour + 20, // minute + 59, // second + ms, // milliseconds + 840 // timezone + ); + // expected success + + assertEquals(calendar2.getMillisecond(), ms); + } + + @DataProvider(name = "invalid-milliseconds") + public Object[][] getInvalidMilliSeconds() { + return new Object[][] { { -1 }, { 1001 } }; + } + + /* + * Test DatatypeFactory.newXMLGregorianCalendar(..). + * + * Bug # 5049592 IllegalArgumentException is thrown if milliseconds < 0 or > + * 1001. + * + */ + @Test(expectedExceptions = IllegalArgumentException.class, dataProvider = "invalid-milliseconds") + public void checkNewCalendarNeg(int milliseconds) { + // invalid milliseconds + datatypeFactory.newXMLGregorianCalendar(2004, // year + 6, // month + 2, // day + 19, // hour + 20, // minute + 59, // second + milliseconds, // milliseconds + 840 // timezone + ); + } + + /* + * Test DatatypeFactory.newXMLGregorianCalendarTime(..). + * + * Bug # 5049592 IllegalArgumentException is thrown if milliseconds < 0 or > + * 1001. + * + */ + @Test(expectedExceptions = IllegalArgumentException.class, dataProvider = "invalid-milliseconds") + public void checkNewTimeNeg(int milliseconds) { + // invalid milliseconds + datatypeFactory.newXMLGregorianCalendarTime(19, // hour + 20, // minute + 59, // second + milliseconds, // milliseconds + 840 // timezone + ); + } + + @DataProvider(name = "data-for-add") + public Object[][] getDataForAdd() { + return new Object[][] { + //calendar1, calendar2, duration + { "1999-12-31T00:00:00Z", "2000-01-01T00:00:00Z", "P1D" }, + { "2000-12-31T00:00:00Z", "2001-01-01T00:00:00Z", "P1D" }, + { "1998-12-31T00:00:00Z", "1999-01-01T00:00:00Z", "P1D" }, + { "2001-12-31T00:00:00Z", "2002-01-01T00:00:00Z", "P1D" }, + { "2003-04-11T00:00:00Z", "2003-04-12T00:00:00Z", "P1D" }, + { "2003-04-11T00:00:00Z", "2003-04-14T00:00:00Z", "P3D" }, + { "2003-04-30T00:00:00Z", "2003-05-01T00:00:00Z", "P1D" }, + { "2003-02-28T00:00:00Z", "2003-03-01T00:00:00Z", "P1D" }, + { "2000-02-29T00:00:00Z", "2000-03-01T00:00:00Z", "P1D" }, + { "2000-02-28T00:00:00Z", "2000-02-29T00:00:00Z", "P1D" }, + { "1998-01-11T00:00:00Z", "1998-04-11T00:00:00Z", "P90D" }, + { "1999-05-11T00:00:00Z", "2002-05-11T00:00:00Z", "P1096D" }}; + } + + /* + * Test XMLGregorianCalendar.add(Duration). + * + */ + @Test(dataProvider = "data-for-add") + public void checkAddDays(String cal1, String cal2, String dur) { + + XMLGregorianCalendar calendar1 = datatypeFactory.newXMLGregorianCalendar(cal1); + XMLGregorianCalendar calendar2 = datatypeFactory.newXMLGregorianCalendar(cal2); + + Duration duration = datatypeFactory.newDuration(dur); + + XMLGregorianCalendar calendar1Clone = (XMLGregorianCalendar)calendar1.clone(); + + calendar1Clone.add(duration); + assertEquals(calendar1Clone, calendar2); + + calendar2.add(duration.negate()); + assertEquals(calendar2, calendar1); + + } + + @DataProvider(name = "gMonth") + public Object[][] getGMonth() { + return new Object[][] { + { "2000-02" }, + { "2000-03" }, + { "2018-02" }}; + } + /* + * Test XMLGregorianCalendar#isValid(). for gMonth + * + * Bug # 5041845 + * + */ + @Test(dataProvider = "gMonth") + public void checkIsValid(String month) { + + XMLGregorianCalendar gMonth = datatypeFactory.newXMLGregorianCalendar(month); + gMonth.setYear(null); + Assert.assertTrue(gMonth.isValid(), gMonth.toString() + " should isValid"); + + } + + @DataProvider(name = "lexical01") + public Object[][] getLexicalRepresentForNormalize01() { + return new Object[][] { { "2000-01-16T12:00:00Z" }, { "2000-01-16T12:00:00" } }; + } + + /* + * Test XMLGregorianCalendar#normalize(...). + * + * Bug # 5048932 XMLGregorianCalendar.normalize works + * + */ + @Test(dataProvider = "lexical01") + public void checkNormalize01(String lexical) { + XMLGregorianCalendar lhs = datatypeFactory.newXMLGregorianCalendar(lexical); + lhs.normalize(); + } + + @DataProvider(name = "lexical02") + public Object[][] getLexicalRepresentForNormalize02() { + return new Object[][] { { "2000-01-16T00:00:00.01Z" }, { "2000-01-16T00:00:00.01" }, { "13:20:00" } }; + } + + /* + * Test XMLGregorianCalendar#normalize(...). + * + * Bug # 5064587 XMLGregorianCalendar.normalize shall not change timezone + * + */ + @Test(dataProvider = "lexical02") + public void checkNormalize02(String lexical) { + XMLGregorianCalendar orig = datatypeFactory.newXMLGregorianCalendar(lexical); + XMLGregorianCalendar normalized = datatypeFactory.newXMLGregorianCalendar(lexical).normalize(); + + assertEquals(normalized.getTimezone(), orig.getTimezone()); + assertEquals(normalized.getMillisecond(), orig.getMillisecond()); + } + + /* + * Test XMLGregorianCalendar#toGregorianCalendar( TimeZone timezone, Locale + * aLocale, XMLGregorianCalendar defaults) + * + * Bug # 5040542 the defaults XMLGregorianCalendar parameter shall take + * effect + * + */ + @Test + public void checkToGregorianCalendar01() { + + XMLGregorianCalendar time_16_17_18 = datatypeFactory.newXMLGregorianCalendar("16:17:18"); + XMLGregorianCalendar date_2001_02_03 = datatypeFactory.newXMLGregorianCalendar("2001-02-03"); + GregorianCalendar calendar = date_2001_02_03.toGregorianCalendar(null, null, time_16_17_18); + + int year = calendar.get(YEAR); + int minute = calendar.get(MINUTE); + + assertTrue((year == 2001 && minute == 17), " expecting year == 2001, minute == 17" + ", result is year == " + year + ", minute == " + minute); + + + calendar = time_16_17_18.toGregorianCalendar(null, null, date_2001_02_03); + + year = calendar.get(YEAR); + minute = calendar.get(MINUTE); + + assertTrue((year == 2001 && minute == 17), " expecting year == 2001, minute == 17" + ", result is year == " + year + ", minute == " + minute); + + + date_2001_02_03.setMinute(3); + date_2001_02_03.setYear(null); + + XMLGregorianCalendar date_time = datatypeFactory.newXMLGregorianCalendar("2003-04-11T02:13:01Z"); + + calendar = date_2001_02_03.toGregorianCalendar(null, null, date_time); + + year = calendar.get(YEAR); + minute = calendar.get(MINUTE); + int hour = calendar.get(HOUR); + + assertTrue((year == 2003 && hour == 2 && minute == 3), " expecting year == 2003, hour == 2, minute == 3" + ", result is year == " + year + ", hour == " + hour + ", minute == " + minute); + + + } + + /* + * Test XMLGregorianCalendar#toGregorianCalendar( TimeZone timezone, Locale + * aLocale, XMLGregorianCalendar defaults) with the 'defaults' parameter + * being null. + * + * Bug # 5049531 XMLGregorianCalendar.toGregorianCalendar(..) can accept + * 'defaults' is null + * + */ + @Test + public void checkToGregorianCalendar02() { + + XMLGregorianCalendar calendar = datatypeFactory.newXMLGregorianCalendar("2004-05-19T12:00:00+06:00"); + calendar.toGregorianCalendar(TimeZone.getDefault(), Locale.getDefault(), null); + } + + @DataProvider(name = "calendar") + public Object[][] getXMLGregorianCalendarData() { + return new Object[][] { + // year, month, day, hour, minute, second + { 1970, 1, 1, 0, 0, 0 }, // DATETIME + { 1970, 1, 1, undef, undef, undef }, // DATE + { undef, undef, undef, 1, 0, 0 }, // TIME + { 1970, 1, undef, undef, undef, undef }, // GYEARMONTH + { undef, 1, 1, undef, undef, undef }, // GMONTHDAY + { 1970, undef, undef, undef, undef, undef }, // GYEAR + { undef, 1, undef, undef, undef, undef }, // GMONTH + { undef, undef, 1, undef, undef, undef } // GDAY + }; + } + + /* + * Test XMLGregorianCalendar#toString() + * + * Bug # 5049528 + * + */ + @Test(dataProvider = "calendar") + public void checkToStringPos(final int year, final int month, final int day, final int hour, final int minute, final int second) { + XMLGregorianCalendar calendar = datatypeFactory.newXMLGregorianCalendar(year, month, day, hour, minute, second, undef, undef); + calendar.toString(); + } + + /* + * Negative Test XMLGregorianCalendar#toString() + * + * Bug # 5049528 XMLGregorianCalendar.toString throws IllegalStateException + * if all parameters are undef + * + */ + @Test(expectedExceptions = IllegalStateException.class) + public void checkToStringNeg() { + XMLGregorianCalendar calendar = datatypeFactory.newXMLGregorianCalendar(undef, undef, undef, undef, undef, undef, undef, undef); + // expected to fail + calendar.toString(); + } + + private final int undef = DatatypeConstants.FIELD_UNDEFINED; + +} From 1b0daecb2651b7bf8e25e93e931e0f68cc568d7f Mon Sep 17 00:00:00 2001 From: Frank Yuan Date: Fri, 30 Jan 2015 19:10:29 -0800 Subject: [PATCH 32/58] 8051710: Convert JAXP function tests: javax.xml.jaxp14.* to jtreg (testng) tests Reviewed-by: lancea, joehw --- .../ptests/FactoryNewInstanceTest.java | 74 ++++ .../ptests/DocumentBuilderFactoryTest.java | 57 +++ .../ptests/SAXFactoryNewInstanceTest.java | 75 ++++ .../XMLEventFactoryNewInstanceTest.java | 76 ++++ .../XMLInputFactoryNewInstanceTest.java | 76 ++++ .../xml/transform/ptests/Bug6384418Test.java | 64 +++ .../xml/transform/ptests/TransformTest.java | 383 ++++++++++++++++++ .../ptests/TransformerFactoryTest.java | 57 ++- .../xml/transform/xmlfiles/dataentry.xsl | 20 + .../javax/xml/transform/xmlfiles/test.xml | 8 + .../validation/ptests/SchemaFactoryTest.java | 57 +++ .../xml/xpath/ptests/XPathFactoryTest.java | 80 +++- .../libs/jaxp/library/JAXPDataProvider.java | 38 ++ 13 files changed, 1062 insertions(+), 3 deletions(-) create mode 100644 jaxp/test/javax/xml/jaxp/functional/javax/xml/datatype/ptests/FactoryNewInstanceTest.java create mode 100644 jaxp/test/javax/xml/jaxp/functional/javax/xml/parsers/ptests/SAXFactoryNewInstanceTest.java create mode 100644 jaxp/test/javax/xml/jaxp/functional/javax/xml/stream/ptests/XMLEventFactoryNewInstanceTest.java create mode 100644 jaxp/test/javax/xml/jaxp/functional/javax/xml/stream/ptests/XMLInputFactoryNewInstanceTest.java create mode 100644 jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/ptests/Bug6384418Test.java create mode 100644 jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/ptests/TransformTest.java create mode 100644 jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/xmlfiles/dataentry.xsl create mode 100644 jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/xmlfiles/test.xml create mode 100644 jaxp/test/javax/xml/jaxp/libs/jaxp/library/JAXPDataProvider.java diff --git a/jaxp/test/javax/xml/jaxp/functional/javax/xml/datatype/ptests/FactoryNewInstanceTest.java b/jaxp/test/javax/xml/jaxp/functional/javax/xml/datatype/ptests/FactoryNewInstanceTest.java new file mode 100644 index 00000000000..48f4a19c8d5 --- /dev/null +++ b/jaxp/test/javax/xml/jaxp/functional/javax/xml/datatype/ptests/FactoryNewInstanceTest.java @@ -0,0 +1,74 @@ +/* + * Copyright (c) 1999, 2015, 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. + */ + +package javax.xml.datatype.ptests; + +import static org.testng.Assert.assertNotNull; + +import javax.xml.datatype.DatatypeConfigurationException; +import javax.xml.datatype.DatatypeFactory; +import javax.xml.datatype.Duration; + +import jaxp.library.JAXPDataProvider; +import jaxp.library.JAXPBaseTest; + +import org.testng.annotations.DataProvider; +import org.testng.annotations.Test; + +/* + * @summary Tests for DatatypeFactory.newInstance(factoryClassName , classLoader) + */ +public class FactoryNewInstanceTest extends JAXPBaseTest { + + private static final String DATATYPE_FACTORY_CLASSNAME = "com.sun.org.apache.xerces.internal.jaxp.datatype.DatatypeFactoryImpl"; + + @DataProvider(name = "parameters") + public Object[][] getValidateParameters() { + return new Object[][] { { DATATYPE_FACTORY_CLASSNAME, null }, { DATATYPE_FACTORY_CLASSNAME, this.getClass().getClassLoader() } }; + } + + /* + * test for DatatypeFactory.newInstance(java.lang.String factoryClassName, + * java.lang.ClassLoader classLoader) factoryClassName points to correct + * implementation of javax.xml.datatype.DatatypeFactory , should return + * newInstance of DatatypeFactory + */ + @Test(dataProvider = "parameters") + public void testNewInstance(String factoryClassName, ClassLoader classLoader) throws DatatypeConfigurationException { + DatatypeFactory dtf = DatatypeFactory.newInstance(DATATYPE_FACTORY_CLASSNAME, null); + Duration duration = dtf.newDuration(true, 1, 1, 1, 1, 1, 1); + assertNotNull(duration); + } + + + /* + * test for DatatypeFactory.newInstance(java.lang.String factoryClassName, + * java.lang.ClassLoader classLoader) factoryClassName is null , should + * throw DatatypeConfigurationException + */ + @Test(expectedExceptions = DatatypeConfigurationException.class, dataProvider = "new-instance-neg", dataProviderClass = JAXPDataProvider.class) + public void testNewInstanceNeg(String factoryClassName, ClassLoader classLoader) throws DatatypeConfigurationException { + DatatypeFactory.newInstance(factoryClassName, classLoader); + } + +} diff --git a/jaxp/test/javax/xml/jaxp/functional/javax/xml/parsers/ptests/DocumentBuilderFactoryTest.java b/jaxp/test/javax/xml/jaxp/functional/javax/xml/parsers/ptests/DocumentBuilderFactoryTest.java index 28ef7b0f226..f2ab3f565f8 100644 --- a/jaxp/test/javax/xml/jaxp/functional/javax/xml/parsers/ptests/DocumentBuilderFactoryTest.java +++ b/jaxp/test/javax/xml/jaxp/functional/javax/xml/parsers/ptests/DocumentBuilderFactoryTest.java @@ -26,27 +26,38 @@ package javax.xml.parsers.ptests; import static org.testng.Assert.assertEquals; import static org.testng.Assert.assertNotNull; import static org.testng.Assert.assertNull; + import java.io.BufferedReader; import java.io.File; import java.io.FileInputStream; import java.io.FilePermission; import java.io.FileReader; + import static javax.xml.XMLConstants.W3C_XML_SCHEMA_NS_URI; + import javax.xml.parsers.DocumentBuilder; import javax.xml.parsers.DocumentBuilderFactory; +import javax.xml.parsers.FactoryConfigurationError; +import javax.xml.parsers.ParserConfigurationException; import javax.xml.parsers.SAXParser; import javax.xml.parsers.SAXParserFactory; + import static javax.xml.parsers.ptests.ParserTestConst.GOLDEN_DIR; import static javax.xml.parsers.ptests.ParserTestConst.XML_DIR; + import javax.xml.transform.Transformer; import javax.xml.transform.TransformerFactory; import javax.xml.transform.dom.DOMSource; import javax.xml.transform.sax.SAXResult; + +import jaxp.library.JAXPDataProvider; import jaxp.library.JAXPFileBaseTest; import static jaxp.library.JAXPTestUtilities.USER_DIR; import static jaxp.library.JAXPTestUtilities.compareWithGold; import static org.testng.Assert.assertFalse; import static org.testng.Assert.assertTrue; + +import org.testng.annotations.DataProvider; import org.testng.annotations.Test; import org.w3c.dom.Document; import org.w3c.dom.Element; @@ -59,6 +70,52 @@ import org.xml.sax.helpers.DefaultHandler; * This checks the methods of DocumentBuilderFactoryImpl. */ public class DocumentBuilderFactoryTest extends JAXPFileBaseTest { + /** + * DocumentBuilderFactory implementation class name. + */ + private static final String DOCUMENT_BUILDER_FACTORY_CLASSNAME = "com.sun.org.apache.xerces.internal.jaxp.DocumentBuilderFactoryImpl"; + + /** + * Provide valid DocumentBuilderFactory instantiation parameters. + * + * @return a data provider contains DocumentBuilderFactory instantiation parameters. + */ + @DataProvider(name = "parameters") + public Object[][] getValidateParameters() { + return new Object[][] { { DOCUMENT_BUILDER_FACTORY_CLASSNAME, null }, { DOCUMENT_BUILDER_FACTORY_CLASSNAME, this.getClass().getClassLoader() } }; + } + + /** + * Test for DocumentBuilderFactory.newInstance(java.lang.String + * factoryClassName, java.lang.ClassLoader classLoader) factoryClassName + * points to correct implementation of + * javax.xml.parsers.DocumentBuilderFactory , should return newInstance of + * DocumentBuilderFactory + * + * @param factoryClassName + * @param classLoader + * @throws ParserConfigurationException + */ + @Test(dataProvider = "parameters") + public void testNewInstance(String factoryClassName, ClassLoader classLoader) throws ParserConfigurationException { + DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance(factoryClassName, classLoader); + DocumentBuilder builder = dbf.newDocumentBuilder(); + assertNotNull(builder); + } + + /** + * test for DocumentBuilderFactory.newInstance(java.lang.String + * factoryClassName, java.lang.ClassLoader classLoader) factoryClassName is + * null , should throw FactoryConfigurationError + * + * @param factoryClassName + * @param classLoader + */ + @Test(expectedExceptions = FactoryConfigurationError.class, dataProvider = "new-instance-neg", dataProviderClass = JAXPDataProvider.class) + public void testNewInstanceNeg(String factoryClassName, ClassLoader classLoader) { + DocumentBuilderFactory.newInstance(factoryClassName, classLoader); + } + /** * Test the default functionality of schema support method. * @throws Exception If any errors occur. diff --git a/jaxp/test/javax/xml/jaxp/functional/javax/xml/parsers/ptests/SAXFactoryNewInstanceTest.java b/jaxp/test/javax/xml/jaxp/functional/javax/xml/parsers/ptests/SAXFactoryNewInstanceTest.java new file mode 100644 index 00000000000..d2f3561af9c --- /dev/null +++ b/jaxp/test/javax/xml/jaxp/functional/javax/xml/parsers/ptests/SAXFactoryNewInstanceTest.java @@ -0,0 +1,75 @@ +/* + * Copyright (c) 1999, 2015, 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. + */ + +package javax.xml.parsers.ptests; + +import static org.testng.Assert.assertNotNull; + +import javax.xml.parsers.FactoryConfigurationError; +import javax.xml.parsers.ParserConfigurationException; +import javax.xml.parsers.SAXParser; +import javax.xml.parsers.SAXParserFactory; + +import jaxp.library.JAXPDataProvider; +import jaxp.library.JAXPBaseTest; + +import org.testng.annotations.DataProvider; +import org.testng.annotations.Test; +import org.xml.sax.SAXException; + +/* + * @summary Tests for SAXParserFactory.newInstance(factoryClassName , classLoader) + */ +public class SAXFactoryNewInstanceTest extends JAXPBaseTest { + + private static final String SAXPARSER_FACTORY_CLASSNAME = "com.sun.org.apache.xerces.internal.jaxp.SAXParserFactoryImpl"; + + @DataProvider(name = "parameters") + public Object[][] getValidateParameters() { + return new Object[][] { { SAXPARSER_FACTORY_CLASSNAME, null }, { SAXPARSER_FACTORY_CLASSNAME, this.getClass().getClassLoader() } }; + } + + /* + * test for SAXParserFactory.newInstance(java.lang.String factoryClassName, + * java.lang.ClassLoader classLoader) factoryClassName points to correct + * implementation of javax.xml.parsers.SAXParserFactory , should return + * newInstance of SAXParserFactory + */ + @Test(dataProvider = "parameters") + public void testNewInstance(String factoryClassName, ClassLoader classLoader) throws ParserConfigurationException, SAXException { + SAXParserFactory spf = SAXParserFactory.newInstance(factoryClassName, classLoader); + SAXParser sp = spf.newSAXParser(); + assertNotNull(sp); + } + + /* + * test for SAXParserFactory.newInstance(java.lang.String factoryClassName, + * java.lang.ClassLoader classLoader) factoryClassName is null , should + * throw FactoryConfigurationError + */ + @Test(expectedExceptions = FactoryConfigurationError.class, dataProvider = "new-instance-neg", dataProviderClass = JAXPDataProvider.class) + public void testNewInstanceNeg(String factoryClassName, ClassLoader classLoader) { + SAXParserFactory.newInstance(factoryClassName, classLoader); + } + +} diff --git a/jaxp/test/javax/xml/jaxp/functional/javax/xml/stream/ptests/XMLEventFactoryNewInstanceTest.java b/jaxp/test/javax/xml/jaxp/functional/javax/xml/stream/ptests/XMLEventFactoryNewInstanceTest.java new file mode 100644 index 00000000000..eb991f071b2 --- /dev/null +++ b/jaxp/test/javax/xml/jaxp/functional/javax/xml/stream/ptests/XMLEventFactoryNewInstanceTest.java @@ -0,0 +1,76 @@ +/* + * Copyright (c) 1999, 2015, 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. + */ + +package javax.xml.stream.ptests; + +import static org.testng.Assert.assertNotNull; + +import javax.xml.stream.XMLEventFactory; + +import jaxp.library.JAXPDataProvider; +import jaxp.library.JAXPBaseTest; + +import org.testng.annotations.DataProvider; +import org.testng.annotations.Test; + +/* + * @summary Tests for XMLEventFactory.newFactory(factoryId , classLoader) + */ +public class XMLEventFactoryNewInstanceTest extends JAXPBaseTest { + + private static final String XMLEVENT_FACTORY_CLASSNAME = "com.sun.xml.internal.stream.events.XMLEventFactoryImpl"; + private static final String XMLEVENT_FACRORY_ID = "javax.xml.stream.XMLEventFactory"; + + @DataProvider(name = "parameters") + public Object[][] getValidateParameters() { + return new Object[][] { { XMLEVENT_FACRORY_ID, null }, { XMLEVENT_FACRORY_ID, this.getClass().getClassLoader() } }; + } + + /* + * test for XMLEventFactory.newFactory(java.lang.String factoryClassName, + * java.lang.ClassLoader classLoader) factoryClassName points to correct + * implementation of javax.xml.stream.XMLEventFactory , should return + * newInstance of XMLEventFactory + */ + @Test(dataProvider = "parameters") + public void testNewFactory(String factoryId, ClassLoader classLoader) { + setSystemProperty(XMLEVENT_FACRORY_ID, XMLEVENT_FACTORY_CLASSNAME); + try { + XMLEventFactory xef = XMLEventFactory.newFactory(factoryId, classLoader); + assertNotNull(xef); + } finally { + setSystemProperty(XMLEVENT_FACRORY_ID, null); + } + } + + /* + * test for XMLEventFactory.newFactory(java.lang.String factoryClassName, + * java.lang.ClassLoader classLoader) factoryClassName is null , should + * throw NullPointerException + */ + @Test(expectedExceptions = NullPointerException.class, dataProvider = "new-instance-neg", dataProviderClass = JAXPDataProvider.class) + public void testNewFactoryNeg(String factoryId, ClassLoader classLoader) { + XMLEventFactory.newFactory(null, null); + } + +} diff --git a/jaxp/test/javax/xml/jaxp/functional/javax/xml/stream/ptests/XMLInputFactoryNewInstanceTest.java b/jaxp/test/javax/xml/jaxp/functional/javax/xml/stream/ptests/XMLInputFactoryNewInstanceTest.java new file mode 100644 index 00000000000..0d5afe322e5 --- /dev/null +++ b/jaxp/test/javax/xml/jaxp/functional/javax/xml/stream/ptests/XMLInputFactoryNewInstanceTest.java @@ -0,0 +1,76 @@ +/* + * Copyright (c) 1999, 2015, 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. + */ + +package javax.xml.stream.ptests; + +import static org.testng.Assert.assertNotNull; + +import javax.xml.stream.XMLInputFactory; + +import jaxp.library.JAXPDataProvider; +import jaxp.library.JAXPBaseTest; + +import org.testng.annotations.DataProvider; +import org.testng.annotations.Test; + +/* + * @summary Tests for XMLInputFactory.newFactory(factoryId , classLoader) + */ +public class XMLInputFactoryNewInstanceTest extends JAXPBaseTest { + + private static final String XMLINPUT_FACTORY_CLASSNAME = "com.sun.xml.internal.stream.XMLInputFactoryImpl"; + private static final String XMLINPUT_FACRORY_ID = "javax.xml.stream.XMLInputFactory"; + + @DataProvider(name = "parameters") + public Object[][] getValidateParameters() { + return new Object[][] { { XMLINPUT_FACRORY_ID, null }, { XMLINPUT_FACRORY_ID, this.getClass().getClassLoader() } }; + } + + /* + * test for XMLInputFactory.newFactory(java.lang.String factoryId, + * java.lang.ClassLoader classLoader) factoryClassName points to correct + * implementation of javax.xml.stream.XMLInputFactory , should return + * newInstance of XMLInputFactory + */ + @Test(dataProvider = "parameters") + public void testNewFactory(String factoryId, ClassLoader classLoader) { + setSystemProperty(XMLINPUT_FACRORY_ID, XMLINPUT_FACTORY_CLASSNAME); + try { + XMLInputFactory xif = XMLInputFactory.newFactory(factoryId, classLoader); + assertNotNull(xif); + } finally { + setSystemProperty(XMLINPUT_FACRORY_ID, null); + } + } + + /* + * test for XMLInputFactory.newFactory(java.lang.String factoryClassName, + * java.lang.ClassLoader classLoader) factoryClassName is null , should + * throw NullPointerException + */ + @Test(expectedExceptions = NullPointerException.class, dataProvider = "new-instance-neg", dataProviderClass = JAXPDataProvider.class) + public void testNewFactoryNeg(String factoryId, ClassLoader classLoader) { + XMLInputFactory.newFactory(factoryId, classLoader); + } + +} diff --git a/jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/ptests/Bug6384418Test.java b/jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/ptests/Bug6384418Test.java new file mode 100644 index 00000000000..bb227d3574c --- /dev/null +++ b/jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/ptests/Bug6384418Test.java @@ -0,0 +1,64 @@ +/* + * Copyright (c) 1999, 2015, 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. + */ + +package javax.xml.transform.ptests; + +import java.io.ByteArrayOutputStream; +import java.io.File; + +import javax.xml.parsers.DocumentBuilder; +import javax.xml.parsers.DocumentBuilderFactory; +import javax.xml.transform.Transformer; +import javax.xml.transform.TransformerFactory; +import javax.xml.transform.dom.DOMSource; +import javax.xml.transform.stream.StreamResult; +import javax.xml.transform.stream.StreamSource; + +import jaxp.library.JAXPFileBaseTest; +import static javax.xml.transform.ptests.TransformerTestConst.XML_DIR; + +import org.testng.annotations.Test; +import org.w3c.dom.Document; + +/* + * @bug 6384418 + * @summary verify the transforming won't throw any exception + */ +public class Bug6384418Test extends JAXPFileBaseTest { + + @Test + public void test() throws Exception { + TransformerFactory tfactory = TransformerFactory.newInstance(); + DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance(); + dbf.setNamespaceAware(true); + DocumentBuilder db = dbf.newDocumentBuilder(); + Document document = db.parse(new File(XML_DIR + "dataentry.xsl")); + DOMSource domSource = new DOMSource(document); + + Transformer transformer = tfactory.newTransformer(domSource); + StreamSource streamSource = new StreamSource(new File(XML_DIR + "test.xml")); + StreamResult streamResult = new StreamResult(new ByteArrayOutputStream()); + transformer.transform(streamSource, streamResult); + } + +} diff --git a/jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/ptests/TransformTest.java b/jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/ptests/TransformTest.java new file mode 100644 index 00000000000..417f6411b11 --- /dev/null +++ b/jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/ptests/TransformTest.java @@ -0,0 +1,383 @@ +/* + * Copyright (c) 1999, 2015, 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. + */ + +package javax.xml.transform.ptests; + +import static javax.xml.transform.ptests.TransformerTestConst.XML_DIR; + +import java.io.BufferedWriter; +import java.io.ByteArrayInputStream; +import java.io.ByteArrayOutputStream; +import java.io.IOException; +import java.io.InputStream; +import java.io.OutputStream; +import java.io.OutputStreamWriter; +import java.nio.file.Files; +import java.nio.file.Paths; +import java.util.function.Supplier; + +import javax.xml.parsers.DocumentBuilder; +import javax.xml.parsers.DocumentBuilderFactory; +import javax.xml.parsers.ParserConfigurationException; +import javax.xml.stream.XMLEventWriter; +import javax.xml.stream.XMLInputFactory; +import javax.xml.stream.XMLOutputFactory; +import javax.xml.stream.XMLStreamException; +import javax.xml.stream.XMLStreamReader; +import javax.xml.stream.XMLStreamWriter; +import javax.xml.transform.Result; +import javax.xml.transform.Source; +import javax.xml.transform.Transformer; +import javax.xml.transform.TransformerConfigurationException; +import javax.xml.transform.TransformerFactory; +import javax.xml.transform.dom.DOMSource; +import javax.xml.transform.sax.SAXResult; +import javax.xml.transform.sax.SAXSource; +import javax.xml.transform.stax.StAXResult; +import javax.xml.transform.stax.StAXSource; +import javax.xml.transform.stream.StreamResult; +import javax.xml.transform.stream.StreamSource; + +import jaxp.library.JAXPFileBaseTest; + +import org.testng.annotations.BeforeClass; +import org.testng.annotations.DataProvider; +import org.testng.annotations.Test; +import org.w3c.dom.Document; +import org.xml.sax.Attributes; +import org.xml.sax.ContentHandler; +import org.xml.sax.InputSource; +import org.xml.sax.Locator; +import org.xml.sax.SAXException; + +/* + * @summary Tests for variable combination of Transformer.transform(Source, Result) + */ +@Test(singleThreaded = true) +public class TransformTest extends JAXPFileBaseTest { + + /* + * Initialize the share objects. + */ + @BeforeClass + public void setup() throws Exception { + ifac = XMLInputFactory.newInstance(); + ofac = XMLOutputFactory.newInstance(); + tfac = TransformerFactory.newInstance(); + + DocumentBuilderFactory dbf = DocumentBuilderFactory.newInstance(); + dbf.setNamespaceAware(true); + db = dbf.newDocumentBuilder(); + + xml = Files.readAllBytes(Paths.get(XML_DIR + "cities.xml")); + template = Files.readAllBytes(Paths.get(XML_DIR + "cities.xsl")); + + xmlDoc = db.parse(xmlInputStream()); + } + + @DataProvider(name = "input-provider") + public Object[][] prepareTestCombination() throws Exception { + + Supplier staxStreamSource = () -> new StAXSource(getXMLStreamReader()); + Supplier staxEventSource = this::getStAXEventSource; + Supplier domSource = () -> new DOMSource(xmlDoc); + Supplier saxSource = () -> new SAXSource(new InputSource(xmlInputStream())); + Supplier streamSource = () -> new StreamSource(xmlInputStream()); + + Supplier staxStreamResult = () -> new StAXResult(getXMLStreamWriter()); + Supplier staxEventResult = () -> new StAXResult(getXMLEventWriter()); + Supplier saxResult = this::getHandlerSAXResult; + Supplier streamResult = () -> new StreamResult(transOutputStream()); + + Transformer domTemplateTransformer = createTransformer(getDomTemplate()); + Transformer saxTemplateTransformer = createTransformer(getSAXTemplate()); + Transformer streamTemplateTransformer = createTransformer(getStreamTemplate()); + Transformer noTemplateTransformer = createTransformer(null); + Transformer staxStreamTemplateTransformer = createTransformer(getStAXStreamTemplate()); + Transformer staxEventTemplateTransformer = createTransformer(getStAXEventTemplate()); + + return new Object[][] { + // StAX Stream + { staxStreamSource, staxStreamResult, domTemplateTransformer }, + { staxStreamSource, staxStreamResult, saxTemplateTransformer }, + { staxStreamSource, staxStreamResult, streamTemplateTransformer }, + { staxStreamSource, staxStreamResult, noTemplateTransformer }, + { staxStreamSource, staxStreamResult, staxStreamTemplateTransformer }, + { staxStreamSource, saxResult, domTemplateTransformer }, + { staxStreamSource, streamResult, domTemplateTransformer }, + { domSource, staxStreamResult, domTemplateTransformer }, + { saxSource, staxStreamResult, domTemplateTransformer }, + { streamSource, staxStreamResult, domTemplateTransformer }, + { staxStreamSource, streamResult, saxTemplateTransformer }, + { domSource, staxStreamResult, saxTemplateTransformer }, + { saxSource, staxStreamResult, saxTemplateTransformer }, + { streamSource, staxStreamResult, saxTemplateTransformer }, + { staxStreamSource, streamResult, streamTemplateTransformer }, + { domSource, staxStreamResult, streamTemplateTransformer }, + { saxSource, staxStreamResult, streamTemplateTransformer }, + { streamSource, staxStreamResult, streamTemplateTransformer }, + // StAX Event + { staxEventSource, staxEventResult, domTemplateTransformer }, + { staxEventSource, staxEventResult, saxTemplateTransformer }, + { staxEventSource, staxEventResult, streamTemplateTransformer }, + { staxEventSource, staxEventResult, noTemplateTransformer }, + { staxEventSource, staxEventResult, staxEventTemplateTransformer }, + { staxEventSource, saxResult, domTemplateTransformer }, + { staxEventSource, streamResult, domTemplateTransformer }, + { domSource, staxEventResult, domTemplateTransformer }, + { saxSource, staxEventResult, domTemplateTransformer }, + { streamSource, staxEventResult, domTemplateTransformer }, + { staxEventSource, streamResult, saxTemplateTransformer }, + { domSource, staxEventResult, saxTemplateTransformer }, + { saxSource, staxEventResult, saxTemplateTransformer }, + { streamSource, staxEventResult, saxTemplateTransformer }, + { staxEventSource, streamResult, streamTemplateTransformer }, + { domSource, staxEventResult, streamTemplateTransformer }, + { saxSource, staxEventResult, streamTemplateTransformer }, + { streamSource, staxEventResult, streamTemplateTransformer } }; + } + + /* + * run Transformer.transform(Source, Result) + */ + @Test(dataProvider = "input-provider") + public void testTransform(Supplier src, Supplier res, Transformer transformer) throws Throwable { + try { + transformer.transform(src.get(), res.get()); + } catch (WrapperException e) { + throw e.getCause(); + } + } + + private InputStream xmlInputStream() { + return new ByteArrayInputStream(xml); + } + + private InputStream templateInputStream() { + return new ByteArrayInputStream(template); + } + + private OutputStream transOutputStream() { + return new ByteArrayOutputStream(xml.length); + } + + private XMLStreamReader getXMLStreamReader() { + try { + return ifac.createXMLStreamReader(xmlInputStream()); + } catch (XMLStreamException e) { + throw new WrapperException(e); + } + } + + private XMLStreamWriter getXMLStreamWriter() { + try { + return ofac.createXMLStreamWriter(transOutputStream()); + } catch (XMLStreamException e) { + throw new WrapperException(e); + } + } + + private StAXSource getStAXEventSource() { + try { + return new StAXSource(ifac.createXMLEventReader(xmlInputStream())); + } catch (XMLStreamException e) { + throw new WrapperException(e); + } + } + + private XMLEventWriter getXMLEventWriter() { + try { + return ofac.createXMLEventWriter(transOutputStream()); + } catch (XMLStreamException e) { + throw new WrapperException(e); + } + } + + private SAXResult getHandlerSAXResult() { + SAXResult res = new SAXResult(); + MyContentHandler myContentHandler = new MyContentHandler(transOutputStream()); + res.setHandler(myContentHandler); + return res; + } + + private Source getDomTemplate() throws SAXException, IOException { + return new DOMSource(db.parse(templateInputStream())); + } + + private Source getSAXTemplate() { + return new SAXSource(new InputSource(templateInputStream())); + } + + private Source getStreamTemplate() { + return new StreamSource(templateInputStream()); + } + + private Source getStAXStreamTemplate() throws XMLStreamException { + return new StAXSource(ifac.createXMLStreamReader(templateInputStream())); + } + + private Source getStAXEventTemplate() throws XMLStreamException { + return new StAXSource(ifac.createXMLEventReader(templateInputStream())); + } + + private Transformer createTransformer(Source templateSource) throws TransformerConfigurationException { + Transformer transformer = (templateSource == null) ? tfac.newTransformer() : tfac.newTransformer(templateSource); + transformer.setOutputProperty("indent", "yes"); + return transformer; + + } + + private static class MyContentHandler implements ContentHandler { + private BufferedWriter bWriter; + + public MyContentHandler(OutputStream os) { + bWriter = new BufferedWriter(new OutputStreamWriter(os)); + } + + public void setDocumentLocator(Locator locator) { + } + + public void startDocument() throws SAXException { + String str = "startDocument"; + try { + bWriter.write(str, 0, str.length()); + bWriter.newLine(); + } catch (IOException e) { + System.out.println("bWriter error"); + } + } + + public void endDocument() throws SAXException { + String str = "endDocument"; + try { + bWriter.write(str, 0, str.length()); + bWriter.newLine(); + bWriter.flush(); + bWriter.close(); + } catch (IOException e) { + System.out.println("bWriter error"); + } + } + + public void startPrefixMapping(String prefix, String uri) throws SAXException { + String str = "startPrefixMapping: " + prefix + ", " + uri; + try { + bWriter.write(str, 0, str.length()); + bWriter.newLine(); + } catch (IOException e) { + System.out.println("bWriter error"); + } + } + + public void endPrefixMapping(String prefix) throws SAXException { + String str = "endPrefixMapping: " + prefix; + try { + bWriter.write(str, 0, str.length()); + bWriter.newLine(); + } catch (IOException e) { + System.out.println("bWriter error"); + } + } + + public void startElement(String namespaceURI, String localName, String qName, Attributes atts) throws SAXException { + StringBuilder str = new StringBuilder("startElement: ").append(namespaceURI).append(", ").append(namespaceURI).append(", ").append(qName).append(" : "); + int n = atts.getLength(); + for (int i = 0; i < n; i++) { + str.append(", ").append(atts.getQName(i)).append(" : ").append(atts.getValue(i)); + } + + try { + bWriter.write(str.toString(), 0, str.length()); + bWriter.newLine(); + } catch (IOException e) { + System.out.println("bWriter error"); + } + } + + public void endElement(String namespaceURI, String localName, String qName) throws SAXException { + String str = "endElement: " + namespaceURI + ", " + namespaceURI + ", " + qName; + try { + bWriter.write(str, 0, str.length()); + bWriter.newLine(); + } catch (IOException e) { + System.out.println("bWriter error"); + } + + } + + public void characters(char ch[], int start, int length) throws SAXException { + String str = new String(ch, start, length); + try { + bWriter.write(str, 0, str.length()); + bWriter.newLine(); + } catch (IOException e) { + System.out.println("bWriter error"); + } + } + + public void ignorableWhitespace(char ch[], int start, int length) throws SAXException { + String str = "ignorableWhitespace"; + try { + bWriter.write(str, 0, str.length()); + bWriter.newLine(); + } catch (IOException e) { + System.out.println("bWriter error"); + } + } + + public void processingInstruction(String target, String data) throws SAXException { + String str = "processingInstruction: " + target + ", " + target; + try { + bWriter.write(str, 0, str.length()); + bWriter.newLine(); + } catch (IOException e) { + System.out.println("bWriter error"); + } + } + + public void skippedEntity(String name) throws SAXException { + String str = "skippedEntity: " + name; + try { + bWriter.write(str, 0, str.length()); + bWriter.newLine(); + } catch (IOException e) { + System.out.println("bWriter error"); + } + } + } + + private static class WrapperException extends RuntimeException { + public WrapperException(Throwable cause) { + super(cause); + } + } + + private XMLInputFactory ifac; + private XMLOutputFactory ofac; + private TransformerFactory tfac; + private DocumentBuilder db; + private byte[] xml; + private byte[] template; + private Document xmlDoc; + +} diff --git a/jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/ptests/TransformerFactoryTest.java b/jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/ptests/TransformerFactoryTest.java index 04c255aceb9..7324b1c144e 100644 --- a/jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/ptests/TransformerFactoryTest.java +++ b/jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/ptests/TransformerFactoryTest.java @@ -23,25 +23,78 @@ package javax.xml.transform.ptests; import java.io.*; -import java.io.FileOutputStream; + import javax.xml.parsers.*; import javax.xml.transform.*; import javax.xml.transform.dom.*; + import static javax.xml.transform.ptests.TransformerTestConst.GOLDEN_DIR; import static javax.xml.transform.ptests.TransformerTestConst.XML_DIR; + import javax.xml.transform.stream.*; + +import jaxp.library.JAXPDataProvider; import jaxp.library.JAXPFileBaseTest; import static jaxp.library.JAXPTestUtilities.USER_DIR; import static jaxp.library.JAXPTestUtilities.compareWithGold; +import static org.testng.Assert.assertNotNull; import static org.testng.Assert.assertTrue; + +import org.testng.annotations.DataProvider; import org.testng.annotations.Test; import org.w3c.dom.*; /** * Class containing the test cases for TransformerFactory API's - * getAssociatedStyleSheet method. + * getAssociatedStyleSheet method and TransformerFactory.newInstance(factoryClassName , classLoader). */ public class TransformerFactoryTest extends JAXPFileBaseTest { + /** + * TransformerFactory implementation class name. + */ + private static final String TRANSFORMER_FACTORY_CLASSNAME = "com.sun.org.apache.xalan.internal.xsltc.trax.TransformerFactoryImpl"; + + /** + * Provide valid TransformerFactory instantiation parameters. + * + * @return a data provider contains TransformerFactory instantiation parameters. + */ + @DataProvider(name = "parameters") + public Object[][] getValidateParameters() { + return new Object[][] { { TRANSFORMER_FACTORY_CLASSNAME, null }, { TRANSFORMER_FACTORY_CLASSNAME, this.getClass().getClassLoader() } }; + } + + /** + * Test for TransformerFactory.newInstance(java.lang.String + * factoryClassName, java.lang.ClassLoader classLoader) factoryClassName + * points to correct implementation of + * javax.xml.transform.TransformerFactory , should return newInstance of + * TransformerFactory + * + * @param factoryClassName + * @param classLoader + * @throws TransformerConfigurationException + */ + @Test(dataProvider = "parameters") + public void testNewInstance(String factoryClassName, ClassLoader classLoader) throws TransformerConfigurationException { + TransformerFactory tf = TransformerFactory.newInstance(factoryClassName, classLoader); + Transformer transformer = tf.newTransformer(); + assertNotNull(transformer); + } + + /** + * Test for TransformerFactory.newInstance(java.lang.String + * factoryClassName, java.lang.ClassLoader classLoader) factoryClassName is + * null , should throw TransformerFactoryConfigurationError + * + * @param factoryClassName + * @param classLoader + */ + @Test(expectedExceptions = TransformerFactoryConfigurationError.class, dataProvider = "new-instance-neg", dataProviderClass = JAXPDataProvider.class) + public void testNewInstanceNeg(String factoryClassName, ClassLoader classLoader) { + TransformerFactory.newInstance(factoryClassName, classLoader); + } + /** * This test case checks for the getAssociatedStylesheet method * of TransformerFactory. diff --git a/jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/xmlfiles/dataentry.xsl b/jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/xmlfiles/dataentry.xsl new file mode 100644 index 00000000000..2cb4c00a35c --- /dev/null +++ b/jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/xmlfiles/dataentry.xsl @@ -0,0 +1,20 @@ + + + + + +
+ +
+ + + + + +
diff --git a/jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/xmlfiles/test.xml b/jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/xmlfiles/test.xml new file mode 100644 index 00000000000..4112c06b17b --- /dev/null +++ b/jaxp/test/javax/xml/jaxp/functional/javax/xml/transform/xmlfiles/test.xml @@ -0,0 +1,8 @@ + + + + TypeOfLifeApp + + + + diff --git a/jaxp/test/javax/xml/jaxp/functional/javax/xml/validation/ptests/SchemaFactoryTest.java b/jaxp/test/javax/xml/jaxp/functional/javax/xml/validation/ptests/SchemaFactoryTest.java index de285f21aa8..f2665263556 100644 --- a/jaxp/test/javax/xml/jaxp/functional/javax/xml/validation/ptests/SchemaFactoryTest.java +++ b/jaxp/test/javax/xml/jaxp/functional/javax/xml/validation/ptests/SchemaFactoryTest.java @@ -46,6 +46,8 @@ import javax.xml.transform.stream.StreamSource; import javax.xml.validation.Schema; import javax.xml.validation.SchemaFactory; +import jaxp.library.JAXPDataProvider; + import org.testng.annotations.BeforeClass; import org.testng.annotations.DataProvider; import org.testng.annotations.Test; @@ -81,6 +83,59 @@ public class SchemaFactoryTest { xml = Files.readAllBytes(Paths.get(XML_DIR + "test.xml")); } + + @DataProvider(name = "parameters") + public Object[][] getValidateParameters() { + return new Object[][] { { W3C_XML_SCHEMA_NS_URI, SCHEMA_FACTORY_CLASSNAME, null }, + { W3C_XML_SCHEMA_NS_URI, SCHEMA_FACTORY_CLASSNAME, this.getClass().getClassLoader() } }; + } + + /* + * test for SchemaFactory.newInstance(java.lang.String schemaLanguage, + * java.lang.String factoryClassName, java.lang.ClassLoader classLoader) + * factoryClassName points to correct implementation of + * javax.xml.validation.SchemaFactory , should return newInstance of + * SchemaFactory + */ + @Test(dataProvider = "parameters") + public void testNewInstance(String schemaLanguage, String factoryClassName, ClassLoader classLoader) throws SAXException { + SchemaFactory sf = SchemaFactory.newInstance(W3C_XML_SCHEMA_NS_URI, SCHEMA_FACTORY_CLASSNAME, null); + Schema schema = sf.newSchema(); + assertNotNull(schema); + } + + /* + * test for SchemaFactory.newInstance(java.lang.String schemaLanguage, + * java.lang.String factoryClassName, java.lang.ClassLoader classLoader) + * factoryClassName is null , should throw IllegalArgumentException + */ + @Test(expectedExceptions = IllegalArgumentException.class, dataProvider = "new-instance-neg", dataProviderClass = JAXPDataProvider.class) + public void testNewInstanceWithNullFactoryClassName(String factoryClassName, ClassLoader classLoader) { + + SchemaFactory.newInstance(W3C_XML_SCHEMA_NS_URI, factoryClassName, classLoader); + } + + /* + * test for SchemaFactory.newInstance(java.lang.String schemaLanguage, + * java.lang.String factoryClassName, java.lang.ClassLoader classLoader) + * schemaLanguage is null , should throw NPE + */ + @Test(expectedExceptions = NullPointerException.class) + public void testNewInstanceWithNullSchemaLanguage() { + SchemaFactory.newInstance(null, SCHEMA_FACTORY_CLASSNAME, this.getClass().getClassLoader()); + } + + /* + * test for SchemaFactory.newInstance(java.lang.String schemaLanguage, + * java.lang.String factoryClassName, java.lang.ClassLoader classLoader) + * schemaLanguage is empty , should throw IllegalArgumentException + */ + @Test(expectedExceptions = IllegalArgumentException.class) + public void testNewInstanceWithEmptySchemaLanguage() { + SchemaFactory.newInstance("", SCHEMA_FACTORY_CLASSNAME, this.getClass().getClassLoader()); + } + + @Test(expectedExceptions = SAXParseException.class) public void testNewSchemaDefault() throws SAXException, IOException { validate(sf.newSchema()); @@ -288,6 +343,8 @@ public class SchemaFactoryTest { private static final String UNRECOGNIZED_NAME = "http://xml.org/sax/features/namespace-prefixes"; + private static final String SCHEMA_FACTORY_CLASSNAME = "com.sun.org.apache.xerces.internal.jaxp.validation.XMLSchemaFactory"; + private SchemaFactory sf; private byte[] xsd1; private byte[] xsd2; diff --git a/jaxp/test/javax/xml/jaxp/functional/javax/xml/xpath/ptests/XPathFactoryTest.java b/jaxp/test/javax/xml/jaxp/functional/javax/xml/xpath/ptests/XPathFactoryTest.java index cc33e0a05cf..a55e1b46554 100644 --- a/jaxp/test/javax/xml/jaxp/functional/javax/xml/xpath/ptests/XPathFactoryTest.java +++ b/jaxp/test/javax/xml/jaxp/functional/javax/xml/xpath/ptests/XPathFactoryTest.java @@ -24,10 +24,16 @@ package javax.xml.xpath.ptests; import static javax.xml.xpath.XPathConstants.DOM_OBJECT_MODEL; + +import javax.xml.xpath.XPath; import javax.xml.xpath.XPathFactory; import javax.xml.xpath.XPathFactoryConfigurationException; + +import jaxp.library.JAXPDataProvider; import jaxp.library.JAXPBaseTest; -import static org.testng.AssertJUnit.assertNotNull; +import static org.testng.Assert.assertNotNull; + +import org.testng.annotations.DataProvider; import org.testng.annotations.Test; /** @@ -44,6 +50,78 @@ public class XPathFactoryTest extends JAXPBaseTest { */ private static final String INVALID_URL = "http://java.sun.com/jaxp/xpath/dom1"; + /** + * XPathFactory implementation class name. + */ + private static final String XPATH_FACTORY_CLASSNAME = "com.sun.org.apache.xpath.internal.jaxp.XPathFactoryImpl"; + + + /** + * Provide valid XPathFactory instantiation parameters. + * + * @return a data provider contains XPathFactory instantiation parameters. + */ + @DataProvider(name = "parameters") + public Object[][] getValidateParameters() { + return new Object[][] { { VALID_URL, XPATH_FACTORY_CLASSNAME, null }, { VALID_URL, XPATH_FACTORY_CLASSNAME, this.getClass().getClassLoader() } }; + } + + /** + * Test for XPathFactory.newInstance(java.lang.String uri, java.lang.String + * factoryClassName, java.lang.ClassLoader classLoader) factoryClassName + * points to correct implementation of javax.xml.xpath.XPathFactory , should + * return newInstance of XPathFactory + * + * @param uri + * @param factoryClassName + * @param classLoader + * @throws XPathFactoryConfigurationException + */ + @Test(dataProvider = "parameters") + public void testNewInstance(String uri, String factoryClassName, ClassLoader classLoader) throws XPathFactoryConfigurationException { + XPathFactory xpf = XPathFactory.newInstance(uri, factoryClassName, classLoader); + XPath xpath = xpf.newXPath(); + assertNotNull(xpath); + } + + /** + * Test for XPathFactory.newInstance(java.lang.String uri, java.lang.String + * factoryClassName, java.lang.ClassLoader classLoader) + * + * @param factoryClassName + * @param classLoader + * @throws XPathFactoryConfigurationException + * is expected when factoryClassName is null + */ + @Test(expectedExceptions = XPathFactoryConfigurationException.class, dataProvider = "new-instance-neg", dataProviderClass = JAXPDataProvider.class) + public void testNewInstanceWithNullFactoryClassName(String factoryClassName, ClassLoader classLoader) throws XPathFactoryConfigurationException { + XPathFactory.newInstance(VALID_URL, factoryClassName, classLoader); + } + + /** + * Test for XPathFactory.newInstance(java.lang.String uri, java.lang.String + * factoryClassName, java.lang.ClassLoader classLoader) uri is null , should + * throw NPE + * + * @throws XPathFactoryConfigurationException + */ + @Test(expectedExceptions = NullPointerException.class) + public void testNewInstanceWithNullUri() throws XPathFactoryConfigurationException { + XPathFactory.newInstance(null, XPATH_FACTORY_CLASSNAME, this.getClass().getClassLoader()); + } + + /** + * Test for XPathFactory.newInstance(java.lang.String uri, java.lang.String + * factoryClassName, java.lang.ClassLoader classLoader) + * + * @throws IllegalArgumentException + * is expected when uri is empty + */ + @Test(expectedExceptions = IllegalArgumentException.class) + public void testNewInstanceWithEmptyUri() throws XPathFactoryConfigurationException { + XPathFactory.newInstance("", XPATH_FACTORY_CLASSNAME, this.getClass().getClassLoader()); + } + /** * Test for constructor - XPathFactory.newInstance(). */ diff --git a/jaxp/test/javax/xml/jaxp/libs/jaxp/library/JAXPDataProvider.java b/jaxp/test/javax/xml/jaxp/libs/jaxp/library/JAXPDataProvider.java new file mode 100644 index 00000000000..d09c3d78b0b --- /dev/null +++ b/jaxp/test/javax/xml/jaxp/libs/jaxp/library/JAXPDataProvider.java @@ -0,0 +1,38 @@ +/* + * Copyright (c) 2015, 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. + */ + +package jaxp.library; + +import org.testng.annotations.DataProvider; + +/** + * Provide invalid parameters for negative testing Factory.newInstance. + */ +public class JAXPDataProvider { + + @DataProvider(name = "new-instance-neg") + public static Object[][] getNewInstanceNeg() { + return new Object[][] { { null, null }, { null, JAXPDataProvider.class.getClassLoader() } }; + } + +} From 8aae81b484e2bf00d68a6a68623d417846dc913f Mon Sep 17 00:00:00 2001 From: Paul Sandoz Date: Mon, 2 Feb 2015 14:18:54 +0100 Subject: [PATCH 33/58] 8048689: Clarify documentation on BaseStream.spliterator Reviewed-by: lancea, chegar --- .../share/classes/java/util/stream/BaseStream.java | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/jdk/src/java.base/share/classes/java/util/stream/BaseStream.java b/jdk/src/java.base/share/classes/java/util/stream/BaseStream.java index 61e74867a19..0328b25cafd 100644 --- a/jdk/src/java.base/share/classes/java/util/stream/BaseStream.java +++ b/jdk/src/java.base/share/classes/java/util/stream/BaseStream.java @@ -79,6 +79,14 @@ public interface BaseStream> *

This is a terminal * operation. * + *

+ * The returned spliterator should report the set of characteristics derived + * from the stream pipeline (namely the characteristics derived from the + * stream source spliterator and the intermediate operations). + * Implementations may report a sub-set of those characteristics. For + * example, it may be too expensive to compute the entire set for some or + * all possible stream pipelines. + * * @return the element spliterator for this stream */ Spliterator spliterator(); From 3b9021981f8b416eb67d2cf5e32fb7313973df96 Mon Sep 17 00:00:00 2001 From: Paul Sandoz Date: Mon, 2 Feb 2015 14:19:00 +0100 Subject: [PATCH 34/58] 8059324: orElseThrow has different signatures for OptionalPrimitive and Optional Reviewed-by: lancea, chegar --- jdk/src/java.base/share/classes/java/util/OptionalDouble.java | 2 +- jdk/src/java.base/share/classes/java/util/OptionalInt.java | 2 +- jdk/src/java.base/share/classes/java/util/OptionalLong.java | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/jdk/src/java.base/share/classes/java/util/OptionalDouble.java b/jdk/src/java.base/share/classes/java/util/OptionalDouble.java index 0efc770fa03..8cd7d9f32da 100644 --- a/jdk/src/java.base/share/classes/java/util/OptionalDouble.java +++ b/jdk/src/java.base/share/classes/java/util/OptionalDouble.java @@ -182,7 +182,7 @@ public final class OptionalDouble { * @throws NullPointerException if no value is present and * {@code exceptionSupplier} is null */ - public double orElseThrow(Supplier exceptionSupplier) throws X { + public double orElseThrow(Supplier exceptionSupplier) throws X { if (isPresent) { return value; } else { diff --git a/jdk/src/java.base/share/classes/java/util/OptionalInt.java b/jdk/src/java.base/share/classes/java/util/OptionalInt.java index 5cb9275242b..d7dc88c7b83 100644 --- a/jdk/src/java.base/share/classes/java/util/OptionalInt.java +++ b/jdk/src/java.base/share/classes/java/util/OptionalInt.java @@ -182,7 +182,7 @@ public final class OptionalInt { * @throws NullPointerException if no value is present and * {@code exceptionSupplier} is null */ - public int orElseThrow(Supplier exceptionSupplier) throws X { + public int orElseThrow(Supplier exceptionSupplier) throws X { if (isPresent) { return value; } else { diff --git a/jdk/src/java.base/share/classes/java/util/OptionalLong.java b/jdk/src/java.base/share/classes/java/util/OptionalLong.java index 589a59fd3ae..540d50a2751 100644 --- a/jdk/src/java.base/share/classes/java/util/OptionalLong.java +++ b/jdk/src/java.base/share/classes/java/util/OptionalLong.java @@ -182,7 +182,7 @@ public final class OptionalLong { * @throws NullPointerException if no value is present and * {@code exceptionSupplier} is null */ - public long orElseThrow(Supplier exceptionSupplier) throws X { + public long orElseThrow(Supplier exceptionSupplier) throws X { if (isPresent) { return value; } else { From dd21d2c4db8295f15c8c5c401c12908e8240c1b6 Mon Sep 17 00:00:00 2001 From: Paul Sandoz Date: Mon, 2 Feb 2015 14:19:12 +0100 Subject: [PATCH 35/58] 8050820: Please add java.util.Optional.stream() to convert Optional to Stream Reviewed-by: alundblad, forax, chegar, jrose --- .../share/classes/java/util/Optional.java | 42 +++++++++--- .../classes/java/util/OptionalDouble.java | 29 ++++++++- .../share/classes/java/util/OptionalInt.java | 29 ++++++++- .../share/classes/java/util/OptionalLong.java | 29 ++++++++- jdk/test/java/util/Optional/Basic.java | 33 ++++++++-- jdk/test/java/util/Optional/BasicDouble.java | 21 ++++++ jdk/test/java/util/Optional/BasicInt.java | 65 ++++++++++++------- jdk/test/java/util/Optional/BasicLong.java | 19 ++++++ 8 files changed, 225 insertions(+), 42 deletions(-) diff --git a/jdk/src/java.base/share/classes/java/util/Optional.java b/jdk/src/java.base/share/classes/java/util/Optional.java index d471058ec16..eb8301b2361 100644 --- a/jdk/src/java.base/share/classes/java/util/Optional.java +++ b/jdk/src/java.base/share/classes/java/util/Optional.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2015, 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 @@ -28,6 +28,7 @@ import java.util.function.Consumer; import java.util.function.Function; import java.util.function.Predicate; import java.util.function.Supplier; +import java.util.stream.Stream; /** * A container object which may or may not contain a non-null value. @@ -155,8 +156,9 @@ public final class Optional { * null */ public void ifPresent(Consumer consumer) { - if (value != null) + if (value != null) { consumer.accept(value); + } } /** @@ -172,10 +174,11 @@ public final class Optional { */ public Optional filter(Predicate predicate) { Objects.requireNonNull(predicate); - if (!isPresent()) + if (!isPresent()) { return this; - else + } else { return predicate.test(value) ? this : empty(); + } } /** @@ -209,9 +212,9 @@ public final class Optional { */ public Optional map(Function mapper) { Objects.requireNonNull(mapper); - if (!isPresent()) + if (!isPresent()) { return empty(); - else { + } else { return Optional.ofNullable(mapper.apply(value)); } } @@ -235,13 +238,36 @@ public final class Optional { */ public Optional flatMap(Function> mapper) { Objects.requireNonNull(mapper); - if (!isPresent()) + if (!isPresent()) { return empty(); - else { + } else { return Objects.requireNonNull(mapper.apply(value)); } } + /** + * If a value is present return a sequential {@link Stream} containing only + * that value, otherwise return an empty {@code Stream}. + * + * @apiNote This method can be used to transform a {@code Stream} of + * optional elements to a {@code Stream} of present value elements: + * + *

{@code
+     *     Stream> os = ..
+     *     Stream s = os.flatMap(Optional::stream)
+     * }
+ * + * @return the optional value as a {@code Stream} + * @since 1.9 + */ + public Stream stream() { + if (!isPresent()) { + return Stream.empty(); + } else { + return Stream.of(value); + } + } + /** * Return the value if present, otherwise return {@code other}. * diff --git a/jdk/src/java.base/share/classes/java/util/OptionalDouble.java b/jdk/src/java.base/share/classes/java/util/OptionalDouble.java index 8cd7d9f32da..e699c53d517 100644 --- a/jdk/src/java.base/share/classes/java/util/OptionalDouble.java +++ b/jdk/src/java.base/share/classes/java/util/OptionalDouble.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2015, 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 @@ -27,6 +27,7 @@ package java.util; import java.util.function.DoubleConsumer; import java.util.function.DoubleSupplier; import java.util.function.Supplier; +import java.util.stream.DoubleStream; /** * A container object which may or may not contain a {@code double} value. @@ -138,8 +139,32 @@ public final class OptionalDouble { * null */ public void ifPresent(DoubleConsumer consumer) { - if (isPresent) + if (isPresent) { consumer.accept(value); + } + } + + /** + * If a value is present return a sequential {@link DoubleStream} containing + * only that value, otherwise return an empty {@code DoubleStream}. + * + * @apiNote This method can be used to transform a {@code Stream} of + * optional doubles to a {@code DoubleStream} of present doubles: + * + *
{@code
+     *     Stream os = ..
+     *     DoubleStream s = os.flatMapToDouble(OptionalDouble::stream)
+     * }
+ * + * @return the optional value as a {@code DoubleStream} + * @since 1.9 + */ + public DoubleStream stream() { + if (isPresent) { + return DoubleStream.of(value); + } else { + return DoubleStream.empty(); + } } /** diff --git a/jdk/src/java.base/share/classes/java/util/OptionalInt.java b/jdk/src/java.base/share/classes/java/util/OptionalInt.java index d7dc88c7b83..d8d9382633d 100644 --- a/jdk/src/java.base/share/classes/java/util/OptionalInt.java +++ b/jdk/src/java.base/share/classes/java/util/OptionalInt.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2015, 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 @@ -27,6 +27,7 @@ package java.util; import java.util.function.IntConsumer; import java.util.function.IntSupplier; import java.util.function.Supplier; +import java.util.stream.IntStream; /** * A container object which may or may not contain a {@code int} value. @@ -138,8 +139,32 @@ public final class OptionalInt { * null */ public void ifPresent(IntConsumer consumer) { - if (isPresent) + if (isPresent) { consumer.accept(value); + } + } + + /** + * If a value is present return a sequential {@link IntStream} containing + * only that value, otherwise return an empty {@code IntStream}. + * + * @apiNote This method can be used to transform a {@code Stream} of + * optional integers to an {@code IntStream} of present integers: + * + *
{@code
+     *     Stream os = ..
+     *     IntStream s = os.flatMapToInt(OptionalInt::stream)
+     * }
+ * + * @return the optional value as an {@code IntStream} + * @since 1.9 + */ + public IntStream stream() { + if (isPresent) { + return IntStream.of(value); + } else { + return IntStream.empty(); + } } /** diff --git a/jdk/src/java.base/share/classes/java/util/OptionalLong.java b/jdk/src/java.base/share/classes/java/util/OptionalLong.java index 540d50a2751..4e2a44873b3 100644 --- a/jdk/src/java.base/share/classes/java/util/OptionalLong.java +++ b/jdk/src/java.base/share/classes/java/util/OptionalLong.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2015, 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 @@ -27,6 +27,7 @@ package java.util; import java.util.function.LongConsumer; import java.util.function.LongSupplier; import java.util.function.Supplier; +import java.util.stream.LongStream; /** * A container object which may or may not contain a {@code long} value. @@ -138,8 +139,32 @@ public final class OptionalLong { * null */ public void ifPresent(LongConsumer consumer) { - if (isPresent) + if (isPresent) { consumer.accept(value); + } + } + + /** + * If a value is present return a sequential {@link LongStream} containing + * only that value, otherwise return an empty {@code LongStream}. + * + * @apiNote This method can be used to transform a {@code Stream} of + * optional longs to a {@code LongStream} of present longs: + * + *
{@code
+     *     Stream os = ..
+     *     LongStream s = os.flatMapToLong(OptionalLong::stream)
+     * }
+ * + * @return the optional value as a {@code LongStream} + * @since 1.9 + */ + public LongStream stream() { + if (isPresent) { + return LongStream.of(value); + } else { + return LongStream.empty(); + } } /** diff --git a/jdk/test/java/util/Optional/Basic.java b/jdk/test/java/util/Optional/Basic.java index dc05ff97a78..f37397d90b9 100644 --- a/jdk/test/java/util/Optional/Basic.java +++ b/jdk/test/java/util/Optional/Basic.java @@ -29,6 +29,7 @@ import java.util.NoSuchElementException; import java.util.Optional; +import java.util.stream.Stream; import static org.testng.Assert.*; import org.testng.annotations.Test; @@ -54,8 +55,8 @@ public class Basic { assertSame(null, empty.orElse(null)); RuntimeException orElse = new RuntimeException() { }; assertSame(Boolean.FALSE, empty.orElse(Boolean.FALSE)); - assertSame(null, empty.orElseGet(()-> null)); - assertSame(Boolean.FALSE, empty.orElseGet(()-> Boolean.FALSE)); + assertSame(null, empty.orElseGet(() -> null)); + assertSame(Boolean.FALSE, empty.orElseGet(() -> Boolean.FALSE)); } @Test(expectedExceptions=NoSuchElementException.class) @@ -104,15 +105,15 @@ public class Basic { try { present.ifPresent(v -> { throw new ObscureException(); }); fail(); - } catch(ObscureException expected) { + } catch (ObscureException expected) { } assertSame(Boolean.TRUE, present.orElse(null)); assertSame(Boolean.TRUE, present.orElse(Boolean.FALSE)); assertSame(Boolean.TRUE, present.orElseGet(null)); - assertSame(Boolean.TRUE, present.orElseGet(()-> null)); - assertSame(Boolean.TRUE, present.orElseGet(()-> Boolean.FALSE)); - assertSame(Boolean.TRUE, present.orElseThrow( null)); + assertSame(Boolean.TRUE, present.orElseGet(() -> null)); + assertSame(Boolean.TRUE, present.orElseGet(() -> Boolean.FALSE)); + assertSame(Boolean.TRUE, present.orElseThrow(null)); assertSame(Boolean.TRUE, present.orElseThrow(ObscureException::new)); } @@ -226,6 +227,26 @@ public class Basic { assertSame(l, fixture); } + @Test(groups = "unit") + public void testStream() { + { + Stream s = Optional.empty().stream(); + assertFalse(s.isParallel()); + + Object[] es = s.toArray(); + assertEquals(es.length, 0); + } + + { + Stream s = Optional.of("Duke").stream(); + assertFalse(s.isParallel()); + + String[] es = s.toArray(String[]::new); + assertEquals(es.length, 1); + assertEquals(es[0], "Duke"); + } + } + private static class ObscureException extends RuntimeException { } diff --git a/jdk/test/java/util/Optional/BasicDouble.java b/jdk/test/java/util/Optional/BasicDouble.java index 841dd459e78..df8ae9e0431 100644 --- a/jdk/test/java/util/Optional/BasicDouble.java +++ b/jdk/test/java/util/Optional/BasicDouble.java @@ -29,6 +29,7 @@ import java.util.NoSuchElementException; import java.util.OptionalDouble; +import java.util.stream.DoubleStream; import static org.testng.Assert.*; import org.testng.annotations.Test; @@ -109,6 +110,26 @@ public class BasicDouble { assertEquals(1.0, present.orElseThrow(ObscureException::new)); } + @Test(groups = "unit") + public void testStream() { + { + DoubleStream s = OptionalDouble.empty().stream(); + assertFalse(s.isParallel()); + + double[] es = s.toArray(); + assertEquals(es.length, 0); + } + + { + DoubleStream s = OptionalDouble.of(42.0).stream(); + assertFalse(s.isParallel()); + + double[] es = s.toArray(); + assertEquals(es.length, 1); + assertEquals(es[0], 42.0); + } + } + private static class ObscureException extends RuntimeException { } diff --git a/jdk/test/java/util/Optional/BasicInt.java b/jdk/test/java/util/Optional/BasicInt.java index fa29f95208a..8f3b2d5d82c 100644 --- a/jdk/test/java/util/Optional/BasicInt.java +++ b/jdk/test/java/util/Optional/BasicInt.java @@ -29,6 +29,7 @@ import java.util.NoSuchElementException; import java.util.OptionalInt; +import java.util.stream.IntStream; import static org.testng.Assert.*; import org.testng.annotations.Test; @@ -53,36 +54,36 @@ public class BasicInt { assertEquals(2, empty.orElseGet(()-> 2)); } - @Test(expectedExceptions=NoSuchElementException.class) - public void testEmptyGet() { - OptionalInt empty = OptionalInt.empty(); + @Test(expectedExceptions=NoSuchElementException.class) + public void testEmptyGet() { + OptionalInt empty = OptionalInt.empty(); - int got = empty.getAsInt(); - } + int got = empty.getAsInt(); + } - @Test(expectedExceptions=NullPointerException.class) - public void testEmptyOrElseGetNull() { - OptionalInt empty = OptionalInt.empty(); + @Test(expectedExceptions=NullPointerException.class) + public void testEmptyOrElseGetNull() { + OptionalInt empty = OptionalInt.empty(); - int got = empty.orElseGet(null); - } + int got = empty.orElseGet(null); + } - @Test(expectedExceptions=NullPointerException.class) - public void testEmptyOrElseThrowNull() throws Throwable { - OptionalInt empty = OptionalInt.empty(); + @Test(expectedExceptions=NullPointerException.class) + public void testEmptyOrElseThrowNull() throws Throwable { + OptionalInt empty = OptionalInt.empty(); - int got = empty.orElseThrow(null); - } + int got = empty.orElseThrow(null); + } - @Test(expectedExceptions=ObscureException.class) - public void testEmptyOrElseThrow() throws Exception { - OptionalInt empty = OptionalInt.empty(); + @Test(expectedExceptions=ObscureException.class) + public void testEmptyOrElseThrow() throws Exception { + OptionalInt empty = OptionalInt.empty(); - int got = empty.orElseThrow(ObscureException::new); - } + int got = empty.orElseThrow(ObscureException::new); + } - @Test(groups = "unit") - public void testPresent() { + @Test(groups = "unit") + public void testPresent() { OptionalInt empty = OptionalInt.empty(); OptionalInt present = OptionalInt.of(1); @@ -109,6 +110,26 @@ public class BasicInt { assertEquals(1, present.orElseThrow(ObscureException::new)); } + @Test(groups = "unit") + public void testStream() { + { + IntStream s = OptionalInt.empty().stream(); + assertFalse(s.isParallel()); + + int[] es = s.toArray(); + assertEquals(es.length, 0); + } + + { + IntStream s = OptionalInt.of(42).stream(); + assertFalse(s.isParallel()); + + int[] es = OptionalInt.of(42).stream().toArray(); + assertEquals(es.length, 1); + assertEquals(es[0], 42); + } + } + private static class ObscureException extends RuntimeException { } diff --git a/jdk/test/java/util/Optional/BasicLong.java b/jdk/test/java/util/Optional/BasicLong.java index 85bf689ad46..39b29375cac 100644 --- a/jdk/test/java/util/Optional/BasicLong.java +++ b/jdk/test/java/util/Optional/BasicLong.java @@ -29,6 +29,7 @@ import java.util.NoSuchElementException; import java.util.OptionalLong; +import java.util.stream.LongStream; import static org.testng.Assert.*; import org.testng.annotations.Test; @@ -109,6 +110,24 @@ public class BasicLong { assertEquals(1, present.orElseThrow(ObscureException::new)); } + @Test(groups = "unit") + public void testStream() { + { + LongStream s = OptionalLong.empty().stream(); + + long[] es = s.toArray(); + assertEquals(es.length, 0); + } + + { + LongStream s = OptionalLong.of(42L).stream(); + + long[] es = s.toArray(); + assertEquals(es.length, 1); + assertEquals(es[0], 42L); + } + } + private static class ObscureException extends RuntimeException { } From c007f0f7d405393800a5ff0b09e334ba8c66d695 Mon Sep 17 00:00:00 2001 From: Paul Sandoz Date: Mon, 2 Feb 2015 14:21:32 +0100 Subject: [PATCH 36/58] 8072030: Race condition in ThenComposeExceptionTest.java Reviewed-by: chegar --- .../ThenComposeExceptionTest.java | 25 +++++++++++++------ 1 file changed, 18 insertions(+), 7 deletions(-) diff --git a/jdk/test/java/util/concurrent/CompletableFuture/ThenComposeExceptionTest.java b/jdk/test/java/util/concurrent/CompletableFuture/ThenComposeExceptionTest.java index ad77c863d9f..877bbb9fd4d 100644 --- a/jdk/test/java/util/concurrent/CompletableFuture/ThenComposeExceptionTest.java +++ b/jdk/test/java/util/concurrent/CompletableFuture/ThenComposeExceptionTest.java @@ -36,7 +36,7 @@ import java.util.function.Consumer; /** * @test - * @bug 8068432 + * @bug 8068432 8072030 * @run testng ThenComposeExceptionTest * @summary Test that CompletableFuture.thenCompose works correctly if the * composing future completes exceptionally @@ -92,7 +92,8 @@ public class ThenComposeExceptionTest { Assert.assertNotSame(f_thenCompose, fe, "Composed CompletableFuture returned directly"); AtomicReference eOnWhenComplete = new AtomicReference<>(); - f_thenCompose.whenComplete((r, e) -> eOnWhenComplete.set(e)); + CompletableFuture f_whenComplete = f_thenCompose. + whenComplete((r, e) -> eOnWhenComplete.set(e)); afterAction.accept(fe); @@ -103,10 +104,20 @@ public class ThenComposeExceptionTest { catch (Throwable t) { eOnJoined = t; } - - Assert.assertTrue(eOnWhenComplete.get() instanceof CompletionException, - "Incorrect exception reported on whenComplete"); Assert.assertTrue(eOnJoined instanceof CompletionException, - "Incorrect exception reported when joined"); + "Incorrect exception reported when joined on thenCompose: " + eOnJoined); + + // Need to wait for f_whenComplete to complete to avoid + // race condition when updating eOnWhenComplete + eOnJoined = null; + try { + f_whenComplete.join(); + } catch (Throwable t) { + eOnJoined = t; + } + Assert.assertTrue(eOnJoined instanceof CompletionException, + "Incorrect exception reported when joined on whenComplete: " + eOnJoined); + Assert.assertTrue(eOnWhenComplete.get() instanceof CompletionException, + "Incorrect exception passed to whenComplete: " + eOnWhenComplete.get()); } -} +} \ No newline at end of file From 887b823a0d4340e3367abcbb4f66d9850ec07961 Mon Sep 17 00:00:00 2001 From: Joe Darcy Date: Mon, 2 Feb 2015 12:35:18 -0800 Subject: [PATCH 37/58] 8072135: Add javax/xml/ws/8046817/GenerateEnumSchema.java to the problem list Reviewed-by: rriggs --- jdk/test/ProblemList.txt | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/jdk/test/ProblemList.txt b/jdk/test/ProblemList.txt index fde49fa1203..415f696fb0d 100644 --- a/jdk/test/ProblemList.txt +++ b/jdk/test/ProblemList.txt @@ -1,6 +1,6 @@ ########################################################################### # -# Copyright (c) 2009, 2014, Oracle and/or its affiliates. All rights reserved. +# Copyright (c) 2009, 2015, 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 @@ -162,6 +162,9 @@ javax/management/remote/mandatory/notif/NotifReconnectDeadlockTest.java generi # 6988950 demo/jvmti/compiledMethodLoad/CompiledMethodLoadTest.java generic-all +# 8071968 +javax/xml/ws/8046817/GenerateEnumSchema.java windows-all + ############################################################################ # jdk_net From 4f6e5d902d151b2644c90ee5d1fa639dac2373c5 Mon Sep 17 00:00:00 2001 From: Joe Darcy Date: Mon, 2 Feb 2015 16:35:10 -0800 Subject: [PATCH 38/58] 8072371: Add two failing svc tests to the problem list Reviewed-by: lancea --- jdk/test/ProblemList.txt | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/jdk/test/ProblemList.txt b/jdk/test/ProblemList.txt index 415f696fb0d..6ea1bab7805 100644 --- a/jdk/test/ProblemList.txt +++ b/jdk/test/ProblemList.txt @@ -132,6 +132,9 @@ java/lang/ClassLoader/deadlock/GetResource.java generic-all java/lang/instrument/RedefineBigClass.sh generic-all java/lang/instrument/RetransformBigClass.sh generic-all +# 8072130 +java/lang/instrument/BootClassPath/BootClassPathTest.sh macosx-all + ############################################################################ # jdk_management @@ -320,6 +323,9 @@ java/util/spi/ResourceBundleControlProvider/UserDefaultControlTest.java generic- # 8031482 sun/tools/jcmd/TestJcmdSanity.java windows-all +# 8072131 +sun/tools/jmap/heapconfig/JMapHeapConfigTest.java macosx-all + # 8027668 sun/tools/jstatd/TestJstatdDefaults.java generic-all sun/tools/jstatd/TestJstatdServer.java generic-all From bb5e8afcf2afce75b6cdddc8057c28fb682c08b8 Mon Sep 17 00:00:00 2001 From: Kumar Srinivasan Date: Tue, 3 Feb 2015 09:49:27 -0800 Subject: [PATCH 39/58] 8068033: JNI exception pending in jdk/src/share/bin/java.c Reviewed-by: mchung, serb --- jdk/src/java.base/macosx/native/libjli/java_md_macosx.c | 3 ++- jdk/src/java.base/share/native/libjli/java.c | 9 ++------- jdk/src/java.base/share/native/libjli/java.h | 9 ++++++++- 3 files changed, 12 insertions(+), 9 deletions(-) diff --git a/jdk/src/java.base/macosx/native/libjli/java_md_macosx.c b/jdk/src/java.base/macosx/native/libjli/java_md_macosx.c index 802d8742dee..eb432c12e08 100644 --- a/jdk/src/java.base/macosx/native/libjli/java_md_macosx.c +++ b/jdk/src/java.base/macosx/native/libjli/java_md_macosx.c @@ -1,5 +1,5 @@ /* - * Copyright (c) 2012, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2012, 2015, 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 @@ -1059,6 +1059,7 @@ JVMInit(InvocationFunctions* ifn, jlong threadStackSize, void PostJVMInit(JNIEnv *env, jstring mainClass, JavaVM *vm) { jvmInstance = vm; SetMainClassForAWT(env, mainClass); + CHECK_EXCEPTION_RETURN(); ShowSplashScreen(); } diff --git a/jdk/src/java.base/share/native/libjli/java.c b/jdk/src/java.base/share/native/libjli/java.c index 2f1e7e17971..d988ac3cec2 100644 --- a/jdk/src/java.base/share/native/libjli/java.c +++ b/jdk/src/java.base/share/native/libjli/java.c @@ -1,5 +1,5 @@ /* - * Copyright (c) 1995, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1995, 2015, 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 @@ -350,12 +350,6 @@ JLI_Launch(int argc, char ** argv, /* main argc, argc */ } \ } while (JNI_FALSE) -#define CHECK_EXCEPTION_RETURN() \ - do { \ - if ((*env)->ExceptionOccurred(env)) { \ - return; \ - } \ - } while (JNI_FALSE) int JNICALL JavaMain(void * _args) @@ -466,6 +460,7 @@ JavaMain(void * _args) * of the application class. */ PostJVMInit(env, appClass, vm); + CHECK_EXCEPTION_LEAVE(1); /* * The LoadMainClass not only loads the main class, it will also ensure * that the main method's signature is correct, therefore further checking diff --git a/jdk/src/java.base/share/native/libjli/java.h b/jdk/src/java.base/share/native/libjli/java.h index 615b16cd9c3..5cc7608b566 100644 --- a/jdk/src/java.base/share/native/libjli/java.h +++ b/jdk/src/java.base/share/native/libjli/java.h @@ -1,5 +1,5 @@ /* - * Copyright (c) 1998, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1998, 2015, 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 @@ -254,4 +254,11 @@ typedef struct { #define NULL_CHECK(NC_check_pointer) \ NULL_CHECK_RETURN_VALUE(NC_check_pointer, ) +#define CHECK_EXCEPTION_RETURN() \ + do { \ + if ((*env)->ExceptionOccurred(env)) { \ + return; \ + } \ + } while (JNI_FALSE) + #endif /* _JAVA_H_ */ From ddb472a4dcf1f4db87e1f6e7a8c7ee71d380763b Mon Sep 17 00:00:00 2001 From: Roger Riggs Date: Tue, 3 Feb 2015 14:39:57 -0500 Subject: [PATCH 40/58] 8068278: ArrayIndexOutOfBoundsException instead of DateTimeException in j.t.chrono.JapaneseChronology.eraOf() Corrected era range check Reviewed-by: mchung, lancea --- .../classes/java/time/chrono/JapaneseEra.java | 5 ++-- .../tck/java/time/chrono/TCKJapaneseEra.java | 25 ++++++++++++++++++- 2 files changed, 27 insertions(+), 3 deletions(-) diff --git a/jdk/src/java.base/share/classes/java/time/chrono/JapaneseEra.java b/jdk/src/java.base/share/classes/java/time/chrono/JapaneseEra.java index 0c9e3e81e1d..f289fdb1a93 100644 --- a/jdk/src/java.base/share/classes/java/time/chrono/JapaneseEra.java +++ b/jdk/src/java.base/share/classes/java/time/chrono/JapaneseEra.java @@ -195,10 +195,11 @@ public final class JapaneseEra * @throws DateTimeException if the value is invalid */ public static JapaneseEra of(int japaneseEra) { - if (japaneseEra < MEIJI.eraValue || japaneseEra + ERA_OFFSET > KNOWN_ERAS.length) { + int i = ordinal(japaneseEra); + if (i < 0 || i >= KNOWN_ERAS.length) { throw new DateTimeException("Invalid era: " + japaneseEra); } - return KNOWN_ERAS[ordinal(japaneseEra)]; + return KNOWN_ERAS[i]; } /** diff --git a/jdk/test/java/time/tck/java/time/chrono/TCKJapaneseEra.java b/jdk/test/java/time/tck/java/time/chrono/TCKJapaneseEra.java index de83e1d3bc3..8e2d341aaa5 100644 --- a/jdk/test/java/time/tck/java/time/chrono/TCKJapaneseEra.java +++ b/jdk/test/java/time/tck/java/time/chrono/TCKJapaneseEra.java @@ -59,6 +59,7 @@ package tck.java.time.chrono; import static java.time.temporal.ChronoField.ERA; import static org.testng.Assert.assertEquals; import static org.testng.Assert.assertTrue; +import static org.testng.Assert.fail; import java.time.chrono.Era; import java.time.chrono.JapaneseChronology; @@ -69,7 +70,8 @@ import org.testng.annotations.DataProvider; import org.testng.annotations.Test; /** - * Test. + * Tests for JapaneseEra + * @bug 8068278 */ @Test public class TCKJapaneseEra { @@ -84,6 +86,20 @@ public class TCKJapaneseEra { }; } + @DataProvider(name = "InvalidJapaneseEras") + Object[][] data_of_invalid_eras() { + return new Object[][] { + {-2}, + {-3}, + {3}, + {Integer.MIN_VALUE}, + {Integer.MAX_VALUE}, + }; + } + + //----------------------------------------------------------------------- + // JapaneseEra value test + //----------------------------------------------------------------------- @Test(dataProvider="JapaneseEras") public void test_valueOf(JapaneseEra era , String eraName, int eraValue) { assertEquals(era.getValue(), eraValue); @@ -118,4 +134,11 @@ public class TCKJapaneseEra { } } + //----------------------------------------------------------------------- + // JapaneseChronology.INSTANCE.eraOf invalid era test + //----------------------------------------------------------------------- + @Test(dataProvider="InvalidJapaneseEras", expectedExceptions=java.time.DateTimeException.class) + public void test_outofrange(int era) { + JapaneseChronology.INSTANCE.eraOf(era); + } } From 778c94c9f319052db48d254a8fa87d6ac7309bbc Mon Sep 17 00:00:00 2001 From: Mandy Chung Date: Tue, 3 Feb 2015 14:08:55 -0800 Subject: [PATCH 41/58] 8069551: Move java.security.acl from compact3 to java.base Reviewed-by: alanb, mullan, wetmore --- common/bin/unshuffle_list.txt | 4 ++-- modules.xml | 11 +++-------- 2 files changed, 5 insertions(+), 10 deletions(-) diff --git a/common/bin/unshuffle_list.txt b/common/bin/unshuffle_list.txt index 0219bd7b005..eba22894ea8 100644 --- a/common/bin/unshuffle_list.txt +++ b/common/bin/unshuffle_list.txt @@ -123,6 +123,7 @@ jdk/src/java.base/share/classes/java/lang/reflect : jdk/src/share/classes/java/l jdk/src/java.base/share/classes/java/math : jdk/src/share/classes/java/math jdk/src/java.base/share/classes/java/net : jdk/src/share/classes/java/net jdk/src/java.base/share/classes/java/nio : jdk/src/share/classes/java/nio +jdk/src/java.base/share/classes/java/security/acl : jdk/src/share/classes/java/security/acl jdk/src/java.base/share/classes/java/security/cert : jdk/src/share/classes/java/security/cert jdk/src/java.base/share/classes/java/security/interfaces : jdk/src/share/classes/java/security/interfaces jdk/src/java.base/share/classes/java/security : jdk/src/share/classes/java/security @@ -179,6 +180,7 @@ jdk/src/java.base/share/classes/sun/nio/ch : jdk/src/share/classes/sun/nio/ch jdk/src/java.base/share/classes/sun/nio/cs : jdk/src/share/classes/sun/nio/cs jdk/src/java.base/share/classes/sun/nio/fs : jdk/src/share/classes/sun/nio/fs jdk/src/java.base/share/classes/sun/reflect : jdk/src/share/classes/sun/reflect +jdk/src/java.base/share/classes/sun/security/acl : jdk/src/share/classes/sun/security/acl jdk/src/java.base/share/classes/sun/security/action : jdk/src/share/classes/sun/security/action jdk/src/java.base/share/classes/sun/security/internal : jdk/src/share/classes/sun/security/internal jdk/src/java.base/share/classes/sun/security/jca : jdk/src/share/classes/sun/security/jca @@ -1211,8 +1213,6 @@ jdk/src/java.rmi/share/doc/stub/java/rmi/activation : jdk/src/share/doc/stub/jav jdk/src/java.rmi/unix/bin/java-rmi.cgi.sh : jdk/src/solaris/bin/java-rmi.cgi.sh jdk/src/java.scripting/share/classes/javax/script : jdk/src/share/classes/javax/script jdk/src/java.scripting/share/classes/com/sun/tools/script/shell : jdk/src/share/classes/com/sun/tools/script/shell -jdk/src/java.security.acl/share/classes/java/security/acl : jdk/src/share/classes/java/security/acl -jdk/src/java.security.acl/share/classes/sun/security/acl : jdk/src/share/classes/sun/security/acl jdk/src/java.security.jgss/macosx/native/libosxkrb5/nativeccache.c : jdk/src/share/native/sun/security/krb5/nativeccache.c jdk/src/java.security.jgss/macosx/native/libosxkrb5/SCDynamicStoreConfig.m : jdk/src/macosx/native/sun/security/krb5/SCDynamicStoreConfig.m jdk/src/java.security.jgss/share/classes/javax/security/auth/kerberos : jdk/src/share/classes/javax/security/auth/kerberos diff --git a/modules.xml b/modules.xml index c92c3fe37a5..1844706864c 100644 --- a/modules.xml +++ b/modules.xml @@ -107,6 +107,9 @@ java.security + + java.security.acl + java.security.cert @@ -446,7 +449,6 @@ java.management java.naming java.prefs - java.security.acl java.security.jgss java.security.sasl java.sql.rowset @@ -890,13 +892,6 @@ java.xml.bind java.xml.ws - - java.security.acl - java.base - - java.security.acl - - java.security.jgss java.base From 37a1a74efd35afe24ef3aab47d4e7b0560f6b1f7 Mon Sep 17 00:00:00 2001 From: Mandy Chung Date: Tue, 3 Feb 2015 14:09:20 -0800 Subject: [PATCH 42/58] 8069551: Move java.security.acl from compact3 to java.base Reviewed-by: alanb, mullan, wetmore --- jdk/make/src/classes/build/tools/module/boot.modules | 1 - .../share/classes/java/security/acl/Acl.java | 0 .../share/classes/java/security/acl/AclEntry.java | 0 .../share/classes/java/security/acl/AclNotFoundException.java | 0 .../share/classes/java/security/acl/Group.java | 0 .../share/classes/java/security/acl/LastOwnerException.java | 0 .../share/classes/java/security/acl/NotOwnerException.java | 0 .../share/classes/java/security/acl/Owner.java | 0 .../share/classes/java/security/acl/Permission.java | 0 .../share/classes/java/security/acl/package-info.java | 0 .../share/classes/sun/security/acl/AclEntryImpl.java | 0 .../share/classes/sun/security/acl/AclImpl.java | 0 .../share/classes/sun/security/acl/AllPermissionsImpl.java | 0 .../share/classes/sun/security/acl/GroupImpl.java | 0 .../share/classes/sun/security/acl/OwnerImpl.java | 0 .../share/classes/sun/security/acl/PermissionImpl.java | 0 .../share/classes/sun/security/acl/PrincipalImpl.java | 0 .../share/classes/sun/security/acl/WorldGroupImpl.java | 0 18 files changed, 1 deletion(-) rename jdk/src/{java.security.acl => java.base}/share/classes/java/security/acl/Acl.java (100%) rename jdk/src/{java.security.acl => java.base}/share/classes/java/security/acl/AclEntry.java (100%) rename jdk/src/{java.security.acl => java.base}/share/classes/java/security/acl/AclNotFoundException.java (100%) rename jdk/src/{java.security.acl => java.base}/share/classes/java/security/acl/Group.java (100%) rename jdk/src/{java.security.acl => java.base}/share/classes/java/security/acl/LastOwnerException.java (100%) rename jdk/src/{java.security.acl => java.base}/share/classes/java/security/acl/NotOwnerException.java (100%) rename jdk/src/{java.security.acl => java.base}/share/classes/java/security/acl/Owner.java (100%) rename jdk/src/{java.security.acl => java.base}/share/classes/java/security/acl/Permission.java (100%) rename jdk/src/{java.security.acl => java.base}/share/classes/java/security/acl/package-info.java (100%) rename jdk/src/{java.security.acl => java.base}/share/classes/sun/security/acl/AclEntryImpl.java (100%) rename jdk/src/{java.security.acl => java.base}/share/classes/sun/security/acl/AclImpl.java (100%) rename jdk/src/{java.security.acl => java.base}/share/classes/sun/security/acl/AllPermissionsImpl.java (100%) rename jdk/src/{java.security.acl => java.base}/share/classes/sun/security/acl/GroupImpl.java (100%) rename jdk/src/{java.security.acl => java.base}/share/classes/sun/security/acl/OwnerImpl.java (100%) rename jdk/src/{java.security.acl => java.base}/share/classes/sun/security/acl/PermissionImpl.java (100%) rename jdk/src/{java.security.acl => java.base}/share/classes/sun/security/acl/PrincipalImpl.java (100%) rename jdk/src/{java.security.acl => java.base}/share/classes/sun/security/acl/WorldGroupImpl.java (100%) diff --git a/jdk/make/src/classes/build/tools/module/boot.modules b/jdk/make/src/classes/build/tools/module/boot.modules index 9763b9287d4..d51fbfb4913 100644 --- a/jdk/make/src/classes/build/tools/module/boot.modules +++ b/jdk/make/src/classes/build/tools/module/boot.modules @@ -11,7 +11,6 @@ java.naming java.prefs java.rmi java.scripting -java.security.acl java.security.jgss java.security.sasl java.smartcardio diff --git a/jdk/src/java.security.acl/share/classes/java/security/acl/Acl.java b/jdk/src/java.base/share/classes/java/security/acl/Acl.java similarity index 100% rename from jdk/src/java.security.acl/share/classes/java/security/acl/Acl.java rename to jdk/src/java.base/share/classes/java/security/acl/Acl.java diff --git a/jdk/src/java.security.acl/share/classes/java/security/acl/AclEntry.java b/jdk/src/java.base/share/classes/java/security/acl/AclEntry.java similarity index 100% rename from jdk/src/java.security.acl/share/classes/java/security/acl/AclEntry.java rename to jdk/src/java.base/share/classes/java/security/acl/AclEntry.java diff --git a/jdk/src/java.security.acl/share/classes/java/security/acl/AclNotFoundException.java b/jdk/src/java.base/share/classes/java/security/acl/AclNotFoundException.java similarity index 100% rename from jdk/src/java.security.acl/share/classes/java/security/acl/AclNotFoundException.java rename to jdk/src/java.base/share/classes/java/security/acl/AclNotFoundException.java diff --git a/jdk/src/java.security.acl/share/classes/java/security/acl/Group.java b/jdk/src/java.base/share/classes/java/security/acl/Group.java similarity index 100% rename from jdk/src/java.security.acl/share/classes/java/security/acl/Group.java rename to jdk/src/java.base/share/classes/java/security/acl/Group.java diff --git a/jdk/src/java.security.acl/share/classes/java/security/acl/LastOwnerException.java b/jdk/src/java.base/share/classes/java/security/acl/LastOwnerException.java similarity index 100% rename from jdk/src/java.security.acl/share/classes/java/security/acl/LastOwnerException.java rename to jdk/src/java.base/share/classes/java/security/acl/LastOwnerException.java diff --git a/jdk/src/java.security.acl/share/classes/java/security/acl/NotOwnerException.java b/jdk/src/java.base/share/classes/java/security/acl/NotOwnerException.java similarity index 100% rename from jdk/src/java.security.acl/share/classes/java/security/acl/NotOwnerException.java rename to jdk/src/java.base/share/classes/java/security/acl/NotOwnerException.java diff --git a/jdk/src/java.security.acl/share/classes/java/security/acl/Owner.java b/jdk/src/java.base/share/classes/java/security/acl/Owner.java similarity index 100% rename from jdk/src/java.security.acl/share/classes/java/security/acl/Owner.java rename to jdk/src/java.base/share/classes/java/security/acl/Owner.java diff --git a/jdk/src/java.security.acl/share/classes/java/security/acl/Permission.java b/jdk/src/java.base/share/classes/java/security/acl/Permission.java similarity index 100% rename from jdk/src/java.security.acl/share/classes/java/security/acl/Permission.java rename to jdk/src/java.base/share/classes/java/security/acl/Permission.java diff --git a/jdk/src/java.security.acl/share/classes/java/security/acl/package-info.java b/jdk/src/java.base/share/classes/java/security/acl/package-info.java similarity index 100% rename from jdk/src/java.security.acl/share/classes/java/security/acl/package-info.java rename to jdk/src/java.base/share/classes/java/security/acl/package-info.java diff --git a/jdk/src/java.security.acl/share/classes/sun/security/acl/AclEntryImpl.java b/jdk/src/java.base/share/classes/sun/security/acl/AclEntryImpl.java similarity index 100% rename from jdk/src/java.security.acl/share/classes/sun/security/acl/AclEntryImpl.java rename to jdk/src/java.base/share/classes/sun/security/acl/AclEntryImpl.java diff --git a/jdk/src/java.security.acl/share/classes/sun/security/acl/AclImpl.java b/jdk/src/java.base/share/classes/sun/security/acl/AclImpl.java similarity index 100% rename from jdk/src/java.security.acl/share/classes/sun/security/acl/AclImpl.java rename to jdk/src/java.base/share/classes/sun/security/acl/AclImpl.java diff --git a/jdk/src/java.security.acl/share/classes/sun/security/acl/AllPermissionsImpl.java b/jdk/src/java.base/share/classes/sun/security/acl/AllPermissionsImpl.java similarity index 100% rename from jdk/src/java.security.acl/share/classes/sun/security/acl/AllPermissionsImpl.java rename to jdk/src/java.base/share/classes/sun/security/acl/AllPermissionsImpl.java diff --git a/jdk/src/java.security.acl/share/classes/sun/security/acl/GroupImpl.java b/jdk/src/java.base/share/classes/sun/security/acl/GroupImpl.java similarity index 100% rename from jdk/src/java.security.acl/share/classes/sun/security/acl/GroupImpl.java rename to jdk/src/java.base/share/classes/sun/security/acl/GroupImpl.java diff --git a/jdk/src/java.security.acl/share/classes/sun/security/acl/OwnerImpl.java b/jdk/src/java.base/share/classes/sun/security/acl/OwnerImpl.java similarity index 100% rename from jdk/src/java.security.acl/share/classes/sun/security/acl/OwnerImpl.java rename to jdk/src/java.base/share/classes/sun/security/acl/OwnerImpl.java diff --git a/jdk/src/java.security.acl/share/classes/sun/security/acl/PermissionImpl.java b/jdk/src/java.base/share/classes/sun/security/acl/PermissionImpl.java similarity index 100% rename from jdk/src/java.security.acl/share/classes/sun/security/acl/PermissionImpl.java rename to jdk/src/java.base/share/classes/sun/security/acl/PermissionImpl.java diff --git a/jdk/src/java.security.acl/share/classes/sun/security/acl/PrincipalImpl.java b/jdk/src/java.base/share/classes/sun/security/acl/PrincipalImpl.java similarity index 100% rename from jdk/src/java.security.acl/share/classes/sun/security/acl/PrincipalImpl.java rename to jdk/src/java.base/share/classes/sun/security/acl/PrincipalImpl.java diff --git a/jdk/src/java.security.acl/share/classes/sun/security/acl/WorldGroupImpl.java b/jdk/src/java.base/share/classes/sun/security/acl/WorldGroupImpl.java similarity index 100% rename from jdk/src/java.security.acl/share/classes/sun/security/acl/WorldGroupImpl.java rename to jdk/src/java.base/share/classes/sun/security/acl/WorldGroupImpl.java From 13c456e099822526a3eb533522ba46060819b503 Mon Sep 17 00:00:00 2001 From: Daniel Fuchs Date: Wed, 4 Feb 2015 17:12:03 +0100 Subject: [PATCH 43/58] 8072450: 9-dev build failed on elinux-i586 and rlinux-i586 Added LL suffix to constant declaration Reviewed-by: dholmes, coleenp, sla --- hotspot/src/share/vm/prims/jvm.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/hotspot/src/share/vm/prims/jvm.cpp b/hotspot/src/share/vm/prims/jvm.cpp index 2d52d7e0723..8056a37c05e 100644 --- a/hotspot/src/share/vm/prims/jvm.cpp +++ b/hotspot/src/share/vm/prims/jvm.cpp @@ -304,7 +304,7 @@ JVM_END // java.lang.System, but we choose to keep it here so that it stays next // to JVM_CurrentTimeMillis and JVM_NanoTime -const jlong MAX_DIFF_SECS = 0x0100000000; // 2^32 +const jlong MAX_DIFF_SECS = 0x0100000000LL; // 2^32 const jlong MIN_DIFF_SECS = -MAX_DIFF_SECS; // -2^32 JVM_LEAF(jlong, JVM_GetNanoTimeAdjustment(JNIEnv *env, jclass ignored, jlong offset_secs)) From d63bf0c50af741fdd3f6cb11fa53645476d110c7 Mon Sep 17 00:00:00 2001 From: Weijun Wang Date: Thu, 5 Feb 2015 11:42:39 +0800 Subject: [PATCH 44/58] 8071643: sun.security.krb5.KrbApReq.authenticate() is not thread safe Reviewed-by: mullan --- .../classes/sun/security/krb5/KrbApReq.java | 27 +++++++++---------- 1 file changed, 12 insertions(+), 15 deletions(-) diff --git a/jdk/src/java.security.jgss/share/classes/sun/security/krb5/KrbApReq.java b/jdk/src/java.security.jgss/share/classes/sun/security/krb5/KrbApReq.java index a7361abdec1..6dc93773d7a 100644 --- a/jdk/src/java.security.jgss/share/classes/sun/security/krb5/KrbApReq.java +++ b/jdk/src/java.security.jgss/share/classes/sun/security/krb5/KrbApReq.java @@ -60,16 +60,6 @@ public class KrbApReq { private static boolean DEBUG = Krb5.DEBUG; private static final char[] hexConst = "0123456789ABCDEF".toCharArray(); - private static final MessageDigest md; - - static { - try { - md = MessageDigest.getInstance("MD5"); - } catch (NoSuchAlgorithmException ex) { - throw new RuntimeException("Impossible"); - } - } - /** * Constructs an AP-REQ message to send to the peer. * @param tgsCred the Credentials to be used to construct the @@ -99,10 +89,10 @@ public class KrbApReq { * @param tgsCred the Credentials to be used to construct the * AP Request protocol message. * @param mutualRequired Whether mutual authentication is required - * @param useSubkey Whether the subkey is to be used to protect this + * @param useSubKey Whether the subkey is to be used to protect this * specific application session. If this is not set then the * session key from the ticket will be used. - * @param checksum checksum of the application data that accompanies + * @param cksum checksum of the application data that accompanies * the KRB_AP_REQ. * @throws KrbException for any Kerberos protocol specific error * @throws IOException for any IO related errors @@ -142,8 +132,8 @@ public class KrbApReq { * Constructs an AP-REQ message from the bytes received from the * peer. * @param message The message received from the peer - * @param keys EncrtyptionKeys to decrypt the message; - * key selected will depend on etype used to encrypte data + * @param cred KrbAcceptCredential containing keys to decrypt + * the message; key selected will depend on etype used to encrypt data * @throws KrbException for any Kerberos protocol specific error * @throws IOException for any IO related errors * (e.g. socket operations) @@ -311,7 +301,14 @@ public class KrbApReq { if (!authenticator.ctime.inClockSkew()) throw new KrbApErrException(Krb5.KRB_AP_ERR_SKEW); - byte[] hash = md.digest(apReqMessg.authenticator.cipher); + byte[] hash; + try { + hash = MessageDigest.getInstance("MD5") + .digest(apReqMessg.authenticator.cipher); + } catch (NoSuchAlgorithmException ex) { + throw new AssertionError("Impossible"); + } + char[] h = new char[hash.length * 2]; for (int i=0; i>4]; From fa9a534731743b4e031124a02fbf24e905f67fc8 Mon Sep 17 00:00:00 2001 From: Shanliang Jiang Date: Thu, 5 Feb 2015 12:13:45 +0100 Subject: [PATCH 45/58] 8065213: Specify and implement PlatformMBeanProvider for looking for all platform MBeans Reviewed-by: dfuchs, mchung, jbachorik --- .../sun.management.spi.PlatformMBeanProvider | 25 + .../internal/PlatformMBeanProviderImpl.java | 209 ++++++++ .../DefaultPlatformMBeanProvider.java | 503 ++++++++++++++++++ .../lang/management/ManagementFactory.java | 366 ++++++++----- .../lang/management/PlatformComponent.java | 468 ---------------- .../management/ExtendedPlatformComponent.java | 54 -- .../management/ManagementFactoryHelper.java | 3 - .../management/spi/PlatformMBeanProvider.java | 233 ++++++++ 8 files changed, 1191 insertions(+), 670 deletions(-) create mode 100644 jdk/src/java.management/share/classes/META-INF/services/sun.management.spi.PlatformMBeanProvider create mode 100644 jdk/src/java.management/share/classes/com/sun/management/internal/PlatformMBeanProviderImpl.java create mode 100644 jdk/src/java.management/share/classes/java/lang/management/DefaultPlatformMBeanProvider.java delete mode 100644 jdk/src/java.management/share/classes/java/lang/management/PlatformComponent.java delete mode 100644 jdk/src/java.management/share/classes/sun/management/ExtendedPlatformComponent.java create mode 100644 jdk/src/java.management/share/classes/sun/management/spi/PlatformMBeanProvider.java diff --git a/jdk/src/java.management/share/classes/META-INF/services/sun.management.spi.PlatformMBeanProvider b/jdk/src/java.management/share/classes/META-INF/services/sun.management.spi.PlatformMBeanProvider new file mode 100644 index 00000000000..a5611e80f23 --- /dev/null +++ b/jdk/src/java.management/share/classes/META-INF/services/sun.management.spi.PlatformMBeanProvider @@ -0,0 +1,25 @@ +# +# Copyright (c) 2015, 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. Oracle designates this +# particular file as subject to the "Classpath" exception as provided +# by Oracle in the LICENSE file that accompanied this code. +# +# 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. +# +com.sun.management.internal.PlatformMBeanProviderImpl diff --git a/jdk/src/java.management/share/classes/com/sun/management/internal/PlatformMBeanProviderImpl.java b/jdk/src/java.management/share/classes/com/sun/management/internal/PlatformMBeanProviderImpl.java new file mode 100644 index 00000000000..42d41b65992 --- /dev/null +++ b/jdk/src/java.management/share/classes/com/sun/management/internal/PlatformMBeanProviderImpl.java @@ -0,0 +1,209 @@ +/* + * Copyright (c) 2015, 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * 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. + */ +package com.sun.management.internal; + +import java.lang.management.ManagementFactory; +import java.util.ArrayList; +import java.util.Collections; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.Set; +import java.util.stream.Collectors; +import java.util.stream.Stream; +import javax.management.DynamicMBean; +import javax.management.ObjectName; +import sun.management.ManagementFactoryHelper; +import sun.management.spi.PlatformMBeanProvider; + +public final class PlatformMBeanProviderImpl extends PlatformMBeanProvider { + private final List> mxbeanList; + + public PlatformMBeanProviderImpl() { + mxbeanList = Collections.unmodifiableList(init()); + } + + @Override + public List> getPlatformComponentList() { + return mxbeanList; + } + + private List> init() { + ArrayList> initMBeanList = new ArrayList<>(); + /** + * Garbage Collector in the Java virtual machine. + */ + initMBeanList.add(new PlatformComponent() { + private final Set garbageCollectorMXBeanInterfaceNames + = Collections.unmodifiableSet( + Stream.of("java.lang.management.MemoryManagerMXBean", + "java.lang.management.GarbageCollectorMXBean", + "com.sun.management.GarbageCollectorMXBean") + .collect(Collectors.toSet())); + + @Override + public Set> mbeanInterfaces() { + return Stream.of(java.lang.management.MemoryManagerMXBean.class, + java.lang.management.GarbageCollectorMXBean.class, + com.sun.management.GarbageCollectorMXBean.class) + .collect(Collectors.toSet()); + } + + @Override + public Set mbeanInterfaceNames() { + return garbageCollectorMXBeanInterfaceNames; + } + + @Override + public String getObjectNamePattern() { + return ManagementFactory.GARBAGE_COLLECTOR_MXBEAN_DOMAIN_TYPE + ",name=*"; + } + + @Override + public boolean isSingleton() { + return false; // zero or more instances + } + + @Override + public Map nameToMBeanMap() { + List list + = ManagementFactoryHelper.getGarbageCollectorMXBeans();; + Map map; + if (list.isEmpty()) { + map = Collections.emptyMap(); + } else { + map = new HashMap<>(list.size()); + for (java.lang.management.MemoryManagerMXBean gcm : list) { + map.put(gcm.getObjectName().getCanonicalName(), + gcm); + } + } + return map; + } + }); + + /** + * OperatingSystemMXBean + */ + initMBeanList.add(new PlatformComponent() { + private final Set operatingSystemMXBeanInterfaceNames + = Collections.unmodifiableSet( + Stream.of("java.lang.management.OperatingSystemMXBean", + "com.sun.management.OperatingSystemMXBean", + "com.sun.management.UnixOperatingSystemMXBean") + .collect(Collectors.toSet())); + + @Override + public Set> mbeanInterfaces() { + return Stream.of(java.lang.management.OperatingSystemMXBean.class, + com.sun.management.OperatingSystemMXBean.class, + com.sun.management.UnixOperatingSystemMXBean.class) + .collect(Collectors.toSet()); + } + + @Override + public Set mbeanInterfaceNames() { + return operatingSystemMXBeanInterfaceNames; + } + + @Override + public String getObjectNamePattern() { + return ManagementFactory.OPERATING_SYSTEM_MXBEAN_NAME; + } + + @Override + public Map nameToMBeanMap() { + return Collections.singletonMap( + ManagementFactory.OPERATING_SYSTEM_MXBEAN_NAME, + ManagementFactoryHelper.getOperatingSystemMXBean()); + } + }); + + /** + * Diagnostic support for the HotSpot Virtual Machine. + */ + initMBeanList.add(new PlatformComponent() { + private final Set hotSpotDiagnosticMXBeanInterfaceNames = + Collections.unmodifiableSet(Collections.singleton("com.sun.management.HotSpotDiagnosticMXBean")); + + @Override + public Set> mbeanInterfaces() { + return Collections.singleton(com.sun.management.HotSpotDiagnosticMXBean.class); + } + + @Override + public Set mbeanInterfaceNames() { + return hotSpotDiagnosticMXBeanInterfaceNames; + } + + @Override + public String getObjectNamePattern() { + return "com.sun.management:type=HotSpotDiagnostic"; + } + + @Override + public Map nameToMBeanMap() { + return Collections.singletonMap( + "com.sun.management:type=HotSpotDiagnostic", + ManagementFactoryHelper.getDiagnosticMXBean()); + } + }); + + /** + * DynamicMBean + */ + HashMap dynmbeans + = ManagementFactoryHelper.getPlatformDynamicMBeans(); + final Set dynamicMBeanInterfaceNames = + Collections.unmodifiableSet(Collections.singleton("javax.management.DynamicMBean")); + for (Map.Entry e : dynmbeans.entrySet()) { + initMBeanList.add(new PlatformComponent() { + @Override + public Set mbeanInterfaceNames() { + return dynamicMBeanInterfaceNames; + } + + @Override + public Set> mbeanInterfaces() { + return Collections.emptySet(); // DynamicMBean cannot be used to find an MBean by ManagementFactory + } + + @Override + public String getObjectNamePattern() { + return e.getKey().getCanonicalName(); + } + + @Override + public Map nameToMBeanMap() { + return Collections.singletonMap( + e.getKey().getCanonicalName(), + e.getValue()); + } + }); + } + initMBeanList.trimToSize(); + return initMBeanList; + } +} diff --git a/jdk/src/java.management/share/classes/java/lang/management/DefaultPlatformMBeanProvider.java b/jdk/src/java.management/share/classes/java/lang/management/DefaultPlatformMBeanProvider.java new file mode 100644 index 00000000000..128b0ba8afd --- /dev/null +++ b/jdk/src/java.management/share/classes/java/lang/management/DefaultPlatformMBeanProvider.java @@ -0,0 +1,503 @@ +/* + * Copyright (c) 2015, 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * 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. + */ +package java.lang.management; + +import java.util.ArrayList; +import java.util.Collections; +import java.util.HashMap; +import java.util.List; +import java.util.Map; +import java.util.Set; +import java.util.function.Function; +import java.util.stream.Collectors; +import java.util.stream.Stream; +import javax.management.DynamicMBean; +import javax.management.ObjectName; +import sun.management.ManagementFactoryHelper; +import sun.management.spi.PlatformMBeanProvider; + +class DefaultPlatformMBeanProvider extends PlatformMBeanProvider { + private final List> mxbeanList; + + DefaultPlatformMBeanProvider() { + mxbeanList = Collections.unmodifiableList(init()); + } + + @Override + public List> getPlatformComponentList() { + return mxbeanList; + } + + private List> init() { + ArrayList> initMBeanList = new ArrayList<>(); + /** + * Class loading system of the Java virtual machine. + */ + initMBeanList.add(new PlatformComponent() { + private final Set classLoadingInterfaceNames = + Collections.unmodifiableSet(Collections.singleton( + "java.lang.management.ClassLoadingMXBean")); + + @Override + public Set> mbeanInterfaces() { + return Collections.singleton(ClassLoadingMXBean.class); + } + + @Override + public Set mbeanInterfaceNames() { + return classLoadingInterfaceNames; + } + + @Override + public String getObjectNamePattern() { + return ManagementFactory.CLASS_LOADING_MXBEAN_NAME; + } + + @Override + public Map nameToMBeanMap() { + return Collections.singletonMap( + ManagementFactory.CLASS_LOADING_MXBEAN_NAME, + ManagementFactoryHelper.getClassLoadingMXBean()); + } + }); + + /** + * Compilation system of the Java virtual machine. + */ + initMBeanList.add(new PlatformComponent() { + private final Set compilationMXBeanInterfaceNames + = Collections.unmodifiableSet(Collections.singleton( + "java.lang.management.CompilationMXBean")); + + @Override + public Set> mbeanInterfaces() { + return Collections.singleton(CompilationMXBean.class); + } + + @Override + public Set mbeanInterfaceNames() { + return compilationMXBeanInterfaceNames; + } + + @Override + public String getObjectNamePattern() { + return ManagementFactory.COMPILATION_MXBEAN_NAME; + } + + @Override + public Map nameToMBeanMap() { + CompilationMXBean m = ManagementFactoryHelper.getCompilationMXBean(); + if (m == null) { + return Collections.emptyMap(); + } else { + return Collections.singletonMap( + ManagementFactory.COMPILATION_MXBEAN_NAME, + ManagementFactoryHelper.getCompilationMXBean()); + } + } + }); + + /** + * Memory system of the Java virtual machine. + */ + initMBeanList.add(new PlatformComponent() { + private final Set memoryMXBeanInterfaceNames + = Collections.unmodifiableSet(Collections.singleton( + "java.lang.management.MemoryMXBean")); + + @Override + public Set> mbeanInterfaces() { + return Collections.singleton(MemoryMXBean.class); + } + + @Override + public Set mbeanInterfaceNames() { + return memoryMXBeanInterfaceNames; + } + + @Override + public String getObjectNamePattern() { + return ManagementFactory.MEMORY_MXBEAN_NAME; + } + + @Override + public Map nameToMBeanMap() { + return Collections.singletonMap( + ManagementFactory.MEMORY_MXBEAN_NAME, + ManagementFactoryHelper.getMemoryMXBean()); + } + }); + + /** + * Garbage Collector in the Java virtual machine. + */ + initMBeanList.add(new PlatformComponent() { + private final Set garbageCollectorMXBeanInterfaceNames + = Collections.unmodifiableSet( + Stream.of("java.lang.management.MemoryManagerMXBean", + "java.lang.management.GarbageCollectorMXBean") + .collect(Collectors.toSet())); + @Override + public Set> mbeanInterfaces() { + return Stream.of(MemoryManagerMXBean.class, + GarbageCollectorMXBean.class, + com.sun.management.GarbageCollectorMXBean.class).collect(Collectors.toSet()); + } + + @Override + public Set mbeanInterfaceNames() { + return garbageCollectorMXBeanInterfaceNames; + } + + @Override + public String getObjectNamePattern() { + return ManagementFactory.GARBAGE_COLLECTOR_MXBEAN_DOMAIN_TYPE + ",name=*"; + } + + @Override + public boolean isSingleton() { + return false; // zero or more instances + } + + @Override + public Map nameToMBeanMap() { + List list + = ManagementFactoryHelper.getGarbageCollectorMXBeans(); + Map map; + if (list.isEmpty()) { + map = Collections.emptyMap(); + } else { + map = new HashMap<>(list.size()); + for (MemoryManagerMXBean gcm : list) { + map.put(gcm.getObjectName().getCanonicalName(), + gcm); + } + } + return map; + } + + }); + + /** + * Memory manager in the Java virtual machine. + */ + initMBeanList.add(new PlatformComponent() { + private final Set memoryManagerMXBeanInterfaceNames + = Collections.unmodifiableSet(Collections.singleton( + "java.lang.management.MemoryManagerMXBean")); + + @Override + public Set> mbeanInterfaces() { + return Collections.singleton(MemoryManagerMXBean.class); + } + + @Override + public Set mbeanInterfaceNames() { + return memoryManagerMXBeanInterfaceNames; + } + + @Override + public String getObjectNamePattern() { + return ManagementFactory.MEMORY_MANAGER_MXBEAN_DOMAIN_TYPE + ",name=*"; + } + + @Override + public boolean isSingleton() { + return false; // zero or more instances + } + + @Override + public Map nameToMBeanMap() { + List list + = ManagementFactoryHelper.getMemoryManagerMXBeans(); + return list.stream() + .filter(this::isMemoryManager) + .collect(Collectors.toMap( + pmo -> pmo.getObjectName().getCanonicalName(), Function.identity())); + } + + // ManagementFactoryHelper.getMemoryManagerMXBeans() returns all + // memory managers - we need to filter out those that do not match + // the pattern for which we are registered + private boolean isMemoryManager(MemoryManagerMXBean mbean) { + final ObjectName name = mbean.getObjectName(); + return ManagementFactory.MEMORY_MANAGER_MXBEAN_DOMAIN_TYPE.startsWith(name.getDomain()) + && ManagementFactory.MEMORY_MANAGER_MXBEAN_DOMAIN_TYPE.contains( + "type="+name.getKeyProperty("type")); + } + }); + + /** + * Memory pool in the Java virtual machine. + */ + initMBeanList.add(new PlatformComponent() { + private final Set memoryPoolMXBeanInterfaceNames + = Collections.unmodifiableSet(Collections.singleton( + "java.lang.management.MemoryPoolMXBean")); + + @Override + public Set> mbeanInterfaces() { + return Collections.singleton(MemoryPoolMXBean.class); + } + + @Override + public Set mbeanInterfaceNames() { + return memoryPoolMXBeanInterfaceNames; + } + + @Override + public String getObjectNamePattern() { + return ManagementFactory.MEMORY_POOL_MXBEAN_DOMAIN_TYPE + ",name=*"; + } + + @Override + public boolean isSingleton() { + return false; // zero or more instances + } + + @Override + public Map nameToMBeanMap() { + List list + = ManagementFactoryHelper.getMemoryPoolMXBeans(); + Map map; + if (list.isEmpty()) { + map = Collections.emptyMap(); + } else { + map = new HashMap<>(list.size()); + for (MemoryPoolMXBean mpm : list) { + map.put(mpm.getObjectName().getCanonicalName(), + mpm); + } + } + return map; + } + }); + + /** + * Runtime system of the Java virtual machine. + */ + initMBeanList.add(new PlatformComponent() { + private final Set runtimeMXBeanInterfaceNames + = Collections.unmodifiableSet(Collections.singleton( + "java.lang.management.RuntimeMXBean")); + + @Override + public Set> mbeanInterfaces() { + return Collections.singleton(RuntimeMXBean.class); + } + + @Override + public Set mbeanInterfaceNames() { + return runtimeMXBeanInterfaceNames; + } + + @Override + public String getObjectNamePattern() { + return ManagementFactory.RUNTIME_MXBEAN_NAME; + } + + @Override + public Map nameToMBeanMap() { + return Collections.singletonMap( + ManagementFactory.RUNTIME_MXBEAN_NAME, + ManagementFactoryHelper.getRuntimeMXBean()); + } + }); + + /** + * Threading system of the Java virtual machine. + */ + initMBeanList.add(new PlatformComponent() { + private final Set threadMXBeanInterfaceNames + = Collections.unmodifiableSet(Collections.singleton( + "java.lang.management.ThreadMXBean")); + + @Override + public Set> mbeanInterfaces() { + return Collections.singleton(ThreadMXBean.class); + } + + @Override + public Set mbeanInterfaceNames() { + return threadMXBeanInterfaceNames; + } + + @Override + public String getObjectNamePattern() { + return ManagementFactory.THREAD_MXBEAN_NAME; + } + + @Override + public Map nameToMBeanMap() { + return Collections.singletonMap( + ManagementFactory.THREAD_MXBEAN_NAME, + ManagementFactoryHelper.getThreadMXBean()); + } + }); + + /** + * Logging facility. + */ + initMBeanList.add(new PlatformComponent() { + private final Set platformLoggingMXBeanInterfaceNames + = Collections.unmodifiableSet(Collections.singleton( + "java.lang.management.PlatformLoggingMXBean")); + + @Override + public Set> mbeanInterfaces() { + return Collections.singleton(PlatformLoggingMXBean.class); + } + + @Override + public Set mbeanInterfaceNames() { + return platformLoggingMXBeanInterfaceNames; + } + + @Override + public String getObjectNamePattern() { + return "java.util.logging:type=Logging"; + } + + @Override + public Map nameToMBeanMap() { + return Collections.singletonMap( + "java.util.logging:type=Logging", + ManagementFactoryHelper.getPlatformLoggingMXBean()); + } + }); + + /** + * Buffer pools. + */ + initMBeanList.add(new PlatformComponent() { + private final Set bufferPoolMXBeanInterfaceNames + = Collections.unmodifiableSet(Collections.singleton( + "java.lang.management.BufferPoolMXBean")); + + @Override + public Set> mbeanInterfaces() { + return Collections.singleton(BufferPoolMXBean.class); + } + + @Override + public Set mbeanInterfaceNames() { + return bufferPoolMXBeanInterfaceNames; + } + + @Override + public String getObjectNamePattern() { + return "java.nio:type=BufferPool,name=*"; + } + + @Override + public boolean isSingleton() { + return false; // zero or more instances + } + + @Override + public Map nameToMBeanMap() { + List list + = ManagementFactoryHelper.getBufferPoolMXBeans(); + Map map; + if (list.isEmpty()) { + map = Collections.emptyMap(); + } else { + map = new HashMap<>(list.size()); + list.stream() + .forEach(mbean -> map.put(mbean.getObjectName().getCanonicalName(),mbean)); + } + return map; + } + }); + + /** + * OperatingSystemMXBean + */ + initMBeanList.add(new PlatformComponent() { + private final Set operatingSystemMXBeanInterfaceNames + = Collections.unmodifiableSet(Collections.singleton( + "java.lang.management.OperatingSystemMXBean")); + + @Override + public Set> mbeanInterfaces() { + return Collections.singleton(OperatingSystemMXBean.class); + } + + @Override + public Set mbeanInterfaceNames() { + return operatingSystemMXBeanInterfaceNames; + } + + @Override + public String getObjectNamePattern() { + return ManagementFactory.OPERATING_SYSTEM_MXBEAN_NAME; + } + + @Override + public Map nameToMBeanMap() { + return Collections.singletonMap( + ManagementFactory.OPERATING_SYSTEM_MXBEAN_NAME, + ManagementFactoryHelper.getOperatingSystemMXBean()); + } + + }); + + /** + * DynamicMBean + */ + HashMap dynmbeans + = ManagementFactoryHelper.getPlatformDynamicMBeans(); + final Set dynamicMBeanInterfaceNames = + Collections.unmodifiableSet(Collections.singleton("javax.management.DynamicMBean")); + for (Map.Entry e : dynmbeans.entrySet()) { + initMBeanList.add(new PlatformComponent() { + @Override + public Set> mbeanInterfaces() { + return Collections.emptySet(); + } + + @Override + public Set mbeanInterfaceNames() { + return dynamicMBeanInterfaceNames; + } + + @Override + public String getObjectNamePattern() { + return e.getKey().getCanonicalName(); + } + + @Override + public Map nameToMBeanMap() { + return Collections.singletonMap( + e.getKey().getCanonicalName(), + e.getValue()); + } + }); + } + + initMBeanList.trimToSize(); + return initMBeanList; + } +} diff --git a/jdk/src/java.management/share/classes/java/lang/management/ManagementFactory.java b/jdk/src/java.management/share/classes/java/lang/management/ManagementFactory.java index e17514227fd..9a0e1ebbd61 100644 --- a/jdk/src/java.management/share/classes/java/lang/management/ManagementFactory.java +++ b/jdk/src/java.management/share/classes/java/lang/management/ManagementFactory.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2003, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2003, 2015, 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 @@ -24,35 +24,40 @@ */ package java.lang.management; +import java.io.FilePermission; +import java.io.IOException; import javax.management.DynamicMBean; import javax.management.MBeanServer; import javax.management.MBeanServerConnection; import javax.management.MBeanServerFactory; import javax.management.MBeanServerPermission; import javax.management.NotificationEmitter; -import javax.management.ObjectInstance; import javax.management.ObjectName; -import javax.management.InstanceAlreadyExistsException; import javax.management.InstanceNotFoundException; import javax.management.MalformedObjectNameException; -import javax.management.MBeanRegistrationException; -import javax.management.NotCompliantMBeanException; import javax.management.StandardEmitterMBean; import javax.management.StandardMBean; import java.util.Collections; import java.util.List; import java.util.Set; -import java.util.HashMap; -import java.util.HashSet; import java.util.Map; import java.security.AccessController; import java.security.Permission; import java.security.PrivilegedAction; import java.security.PrivilegedActionException; import java.security.PrivilegedExceptionAction; +import java.util.ArrayList; +import java.util.Collection; +import java.util.Optional; +import java.util.ServiceLoader; +import java.util.function.Function; +import java.util.stream.Collectors; +import static java.util.stream.Collectors.toMap; +import java.util.stream.Stream; import javax.management.JMX; -import sun.management.ManagementFactoryHelper; -import sun.management.ExtendedPlatformComponent; +import sun.management.Util; +import sun.management.spi.PlatformMBeanProvider; +import sun.management.spi.PlatformMBeanProvider.PlatformComponent; /** * The {@code ManagementFactory} class is a factory class for getting @@ -316,7 +321,7 @@ public class ManagementFactory { * the Java virtual machine. */ public static ClassLoadingMXBean getClassLoadingMXBean() { - return ManagementFactoryHelper.getClassLoadingMXBean(); + return getPlatformMXBean(ClassLoadingMXBean.class); } /** @@ -326,7 +331,7 @@ public class ManagementFactory { * @return a {@link MemoryMXBean} object for the Java virtual machine. */ public static MemoryMXBean getMemoryMXBean() { - return ManagementFactoryHelper.getMemoryMXBean(); + return getPlatformMXBean(MemoryMXBean.class); } /** @@ -336,7 +341,7 @@ public class ManagementFactory { * @return a {@link ThreadMXBean} object for the Java virtual machine. */ public static ThreadMXBean getThreadMXBean() { - return ManagementFactoryHelper.getThreadMXBean(); + return getPlatformMXBean(ThreadMXBean.class); } /** @@ -347,7 +352,7 @@ public class ManagementFactory { */ public static RuntimeMXBean getRuntimeMXBean() { - return ManagementFactoryHelper.getRuntimeMXBean(); + return getPlatformMXBean(RuntimeMXBean.class); } /** @@ -360,7 +365,7 @@ public class ManagementFactory { * no compilation system. */ public static CompilationMXBean getCompilationMXBean() { - return ManagementFactoryHelper.getCompilationMXBean(); + return getPlatformMXBean(CompilationMXBean.class); } /** @@ -371,7 +376,7 @@ public class ManagementFactory { * the Java virtual machine. */ public static OperatingSystemMXBean getOperatingSystemMXBean() { - return ManagementFactoryHelper.getOperatingSystemMXBean(); + return getPlatformMXBean(OperatingSystemMXBean.class); } /** @@ -384,7 +389,7 @@ public class ManagementFactory { * */ public static List getMemoryPoolMXBeans() { - return ManagementFactoryHelper.getMemoryPoolMXBeans(); + return getPlatformMXBeans(MemoryPoolMXBean.class); } /** @@ -397,7 +402,7 @@ public class ManagementFactory { * */ public static List getMemoryManagerMXBeans() { - return ManagementFactoryHelper.getMemoryManagerMXBeans(); + return getPlatformMXBeans(MemoryManagerMXBean.class); } @@ -413,7 +418,7 @@ public class ManagementFactory { * */ public static List getGarbageCollectorMXBeans() { - return ManagementFactoryHelper.getGarbageCollectorMXBeans(); + return getPlatformMXBeans(GarbageCollectorMXBean.class); } private static MBeanServer platformMBeanServer; @@ -467,35 +472,11 @@ public class ManagementFactory { if (platformMBeanServer == null) { platformMBeanServer = MBeanServerFactory.createMBeanServer(); - for (PlatformComponent pc : PlatformComponent.values()) { - List list = - pc.getMXBeans(pc.getMXBeanInterface()); - for (PlatformManagedObject o : list) { - // Each PlatformComponent represents one management - // interface. Some MXBean may extend another one. - // The MXBean instances for one platform component - // (returned by pc.getMXBeans()) might be also - // the MXBean instances for another platform component. - // e.g. com.sun.management.GarbageCollectorMXBean - // - // So need to check if an MXBean instance is registered - // before registering into the platform MBeanServer - if (!platformMBeanServer.isRegistered(o.getObjectName())) { - addMXBean(platformMBeanServer, o); - } - } - } - HashMap dynmbeans = - ManagementFactoryHelper.getPlatformDynamicMBeans(); - for (Map.Entry e : dynmbeans.entrySet()) { - addDynamicMBean(platformMBeanServer, e.getValue(), e.getKey()); - } - for (final PlatformManagedObject o : - ExtendedPlatformComponent.getMXBeans()) { - if (!platformMBeanServer.isRegistered(o.getObjectName())) { - addMXBean(platformMBeanServer, o); - } - } + platformComponents() + .stream() + .filter(PlatformComponent::shouldRegister) + .flatMap(pc -> pc.nameToMBeanMap().entrySet().stream()) + .forEach(entry -> addMXBean(platformMBeanServer, entry.getKey(), entry.getValue())); } return platformMBeanServer; } @@ -600,11 +581,8 @@ public class ManagementFactory { // bootstrap class loader final Class cls = mxbeanInterface; ClassLoader loader = - AccessController.doPrivileged(new PrivilegedAction() { - public ClassLoader run() { - return cls.getClassLoader(); - } - }); + AccessController.doPrivileged( + (PrivilegedAction) () -> cls.getClassLoader()); if (!sun.misc.VM.isSystemDomainLoader(loader)) { throw new IllegalArgumentException(mxbeanName + " is not a platform MXBean"); @@ -619,7 +597,6 @@ public class ManagementFactory { " is not an instance of " + mxbeanInterface); } - final Class[] interfaces; // check if the registered MBean is a notification emitter boolean emitter = connection.isInstanceOf(objName, NOTIF_EMITTER); @@ -661,20 +638,11 @@ public class ManagementFactory { */ public static T getPlatformMXBean(Class mxbeanInterface) { - PlatformComponent pc = PlatformComponent.getPlatformComponent(mxbeanInterface); - if (pc == null) { - T mbean = ExtendedPlatformComponent.getMXBean(mxbeanInterface); - if (mbean != null) { - return mbean; - } - throw new IllegalArgumentException(mxbeanInterface.getName() + - " is not a platform management interface"); - } - if (!pc.isSingleton()) - throw new IllegalArgumentException(mxbeanInterface.getName() + - " can have zero or more than one instances"); + PlatformComponent pc = PlatformMBeanFinder.findSingleton(mxbeanInterface); - return pc.getSingletonMXBean(mxbeanInterface); + List mbeans = pc.getMBeans(mxbeanInterface); + assert mbeans.isEmpty() || mbeans.size() == 1; + return mbeans.isEmpty() ? null : mbeans.get(0); } /** @@ -701,16 +669,19 @@ public class ManagementFactory { */ public static List getPlatformMXBeans(Class mxbeanInterface) { - PlatformComponent pc = PlatformComponent.getPlatformComponent(mxbeanInterface); + // Validates at first the specified interface by finding at least one + // PlatformComponent whose MXBean implements this interface. + // An interface can be implemented by different MBeans, provided by + // different platform components. + PlatformComponent pc = PlatformMBeanFinder.findFirst(mxbeanInterface); if (pc == null) { - T mbean = ExtendedPlatformComponent.getMXBean(mxbeanInterface); - if (mbean != null) { - return Collections.singletonList(mbean); - } - throw new IllegalArgumentException(mxbeanInterface.getName() + - " is not a platform management interface"); + throw new IllegalArgumentException(mxbeanInterface.getName() + + " is not a platform management interface"); } - return Collections.unmodifiableList(pc.getMXBeans(mxbeanInterface)); + + return platformComponents().stream() + .flatMap(p -> p.getMBeans(mxbeanInterface).stream()) + .collect(Collectors.toList()); } /** @@ -753,22 +724,8 @@ public class ManagementFactory { Class mxbeanInterface) throws java.io.IOException { - PlatformComponent pc = PlatformComponent.getPlatformComponent(mxbeanInterface); - if (pc == null) { - T mbean = ExtendedPlatformComponent.getMXBean(mxbeanInterface); - if (mbean != null) { - ObjectName on = mbean.getObjectName(); - return ManagementFactory.newPlatformMXBeanProxy(connection, - on.getCanonicalName(), - mxbeanInterface); - } - throw new IllegalArgumentException(mxbeanInterface.getName() + - " is not a platform management interface"); - } - if (!pc.isSingleton()) - throw new IllegalArgumentException(mxbeanInterface.getName() + - " can have zero or more than one instances"); - return pc.getSingletonMXBean(connection, mxbeanInterface); + PlatformComponent pc = PlatformMBeanFinder.findSingleton(mxbeanInterface); + return newPlatformMXBeanProxy(connection, pc.getObjectNamePattern(), mxbeanInterface); } /** @@ -804,19 +761,56 @@ public class ManagementFactory { Class mxbeanInterface) throws java.io.IOException { - PlatformComponent pc = PlatformComponent.getPlatformComponent(mxbeanInterface); + // Validates at first the specified interface by finding at least one + // PlatformComponent whose MXBean implements this interface. + // An interface can be implemented by different MBeans, provided by + // different platform components. + PlatformComponent pc = PlatformMBeanFinder.findFirst(mxbeanInterface); if (pc == null) { - T mbean = ExtendedPlatformComponent.getMXBean(mxbeanInterface); - if (mbean != null) { - ObjectName on = mbean.getObjectName(); - T proxy = ManagementFactory.newPlatformMXBeanProxy(connection, - on.getCanonicalName(), mxbeanInterface); - return Collections.singletonList(proxy); - } - throw new IllegalArgumentException(mxbeanInterface.getName() + - " is not a platform management interface"); + throw new IllegalArgumentException(mxbeanInterface.getName() + + " is not a platform management interface"); } - return Collections.unmodifiableList(pc.getMXBeans(connection, mxbeanInterface)); + + // Collect all names, eliminate duplicates. + Stream names = Stream.empty(); + for (PlatformComponent p : platformComponents()) { + names = Stream.concat(names, getProxyNames(p, connection, mxbeanInterface)); + } + Set objectNames = names.collect(Collectors.toSet()); + if (objectNames.isEmpty()) return Collections.emptyList(); + + // Map names on proxies. + List proxies = new ArrayList<>(); + for (String name : objectNames) { + proxies.add(newPlatformMXBeanProxy(connection, name, mxbeanInterface)); + } + return proxies; + } + + // Returns a stream containing all ObjectNames of the MBeans represented by + // the specified PlatformComponent and implementing the specified interface. + // If the PlatformComponent is a singleton, the name returned by + // PlatformComponent.getObjectNamePattern() will be used, otherwise + // we will query the specified MBeanServerConnection (conn.queryNames) + // with the pattern returned by PlatformComponent.getObjectNamePattern() + // in order to find the names of matching MBeans. + // In case of singleton, we do not check whether the MBean is registered + // in the connection because the caller "getPlatformMXBeans" will do the check + // when creating a proxy. + private static Stream getProxyNames(PlatformComponent pc, + MBeanServerConnection conn, + Class intf) + throws IOException + { + if (pc.mbeanInterfaceNames().contains(intf.getName())) { + if (pc.isSingleton()) { + return Stream.of(pc.getObjectNamePattern()); + } else { + return conn.queryNames(Util.newObjectName(pc.getObjectNamePattern()), null) + .stream().map(ObjectName::getCanonicalName); + } + } + return Stream.empty(); } /** @@ -835,63 +829,145 @@ public class ManagementFactory { public static Set> getPlatformManagementInterfaces() { - Set> result = - new HashSet<>(); - for (PlatformComponent component: PlatformComponent.values()) { - result.add(component.getMXBeanInterface()); - } - return Collections.unmodifiableSet(result); + return platformComponents() + .stream() + .flatMap(pc -> pc.mbeanInterfaces().stream()) + .filter(clazz -> PlatformManagedObject.class.isAssignableFrom(clazz)) + .map(clazz -> clazz.asSubclass(PlatformManagedObject.class)) + .collect(Collectors.toSet()); } private static final String NOTIF_EMITTER = "javax.management.NotificationEmitter"; - /** - * Registers an MXBean. - */ - private static void addMXBean(final MBeanServer mbs, final PlatformManagedObject pmo) { - // Make DynamicMBean out of MXBean by wrapping it with a StandardMBean + private static void addMXBean(final MBeanServer mbs, String name, final Object pmo) + { try { - AccessController.doPrivileged(new PrivilegedExceptionAction() { - public Void run() throws InstanceAlreadyExistsException, - MBeanRegistrationException, - NotCompliantMBeanException { - final DynamicMBean dmbean; - if (pmo instanceof DynamicMBean) { - dmbean = DynamicMBean.class.cast(pmo); - } else if (pmo instanceof NotificationEmitter) { - dmbean = new StandardEmitterMBean(pmo, null, true, (NotificationEmitter) pmo); - } else { - dmbean = new StandardMBean(pmo, null, true); - } - - mbs.registerMBean(dmbean, pmo.getObjectName()); - return null; + ObjectName oname = ObjectName.getInstance(name); + // Make DynamicMBean out of MXBean by wrapping it with a StandardMBean + AccessController.doPrivileged((PrivilegedExceptionAction) () -> { + final DynamicMBean dmbean; + if (pmo instanceof DynamicMBean) { + dmbean = DynamicMBean.class.cast(pmo); + } else if (pmo instanceof NotificationEmitter) { + dmbean = new StandardEmitterMBean(pmo, null, true, (NotificationEmitter) pmo); + } else { + dmbean = new StandardMBean(pmo, null, true); } + + mbs.registerMBean(dmbean, oname); + return null; }); + } catch (MalformedObjectNameException mone) { + throw new IllegalArgumentException(mone); } catch (PrivilegedActionException e) { throw new RuntimeException(e.getException()); } } - /** - * Registers a DynamicMBean. - */ - private static void addDynamicMBean(final MBeanServer mbs, - final DynamicMBean dmbean, - final ObjectName on) { - try { - AccessController.doPrivileged(new PrivilegedExceptionAction() { - @Override - public Void run() throws InstanceAlreadyExistsException, - MBeanRegistrationException, - NotCompliantMBeanException { - mbs.registerMBean(dmbean, on); - return null; - } - }); - } catch (PrivilegedActionException e) { - throw new RuntimeException(e.getException()); + private static Collection> platformComponents() + { + return PlatformMBeanFinder.getMap().values(); + } + + private static class PlatformMBeanFinder + { + private static final Map> componentMap; + static { + // get all providers + List providers = AccessController.doPrivileged( + (PrivilegedAction>) () -> { + List all = new ArrayList<>(); + ServiceLoader.loadInstalled(PlatformMBeanProvider.class) + .forEach(all::add); + all.add(new DefaultPlatformMBeanProvider()); + return all; + }, null, new FilePermission("<>", "read"), + new RuntimePermission("sun.management.spi.PlatformMBeanProvider")); + + // load all platform components into a map + componentMap = providers.stream() + .flatMap(p -> toPlatformComponentStream(p)) + // The first one wins if multiple PlatformComponents + // with same ObjectName pattern, + .collect(toMap(PlatformComponent::getObjectNamePattern, + Function.identity(), + (p1, p2) -> p1)); + } + + static Map> getMap() { + return componentMap; + } + + // Loads all platform components from a provider into a stream + // Ensures that two different components are not declared with the same + // object name pattern. Throws InternalError if the provider incorrectly + // declares two platform components with the same pattern. + private static Stream> + toPlatformComponentStream(PlatformMBeanProvider provider) + { + return provider.getPlatformComponentList() + .stream() + .collect(toMap(PlatformComponent::getObjectNamePattern, + Function.identity(), + (p1, p2) -> { + throw new InternalError( + p1.getObjectNamePattern() + + " has been used as key for " + p1 + + ", it cannot be reused for " + p2); + })) + .values().stream(); + } + + // Finds the first PlatformComponent whose mbeanInterfaceNames() list + // contains the specified class name. An MBean interface can be implemented + // by different MBeans, provided by different platform components. + // For instance the MemoryManagerMXBean interface is implemented both by + // regular memory managers, and garbage collector MXBeans. This method is + // mainly used to verify that there is at least one PlatformComponent + // which provides an implementation of the desired interface. + static PlatformComponent findFirst(Class mbeanIntf) + { + String name = mbeanIntf.getName(); + Optional> op = getMap().values() + .stream() + .filter(pc -> pc.mbeanInterfaceNames().contains(name)) + .findFirst(); + + if (op.isPresent()) { + return op.get(); + } else { + return null; + } + } + + // Finds a PlatformComponent whose mbeanInterface name list contains + // the specified class name, and make sure that one and only one exists. + static PlatformComponent findSingleton(Class mbeanIntf) + { + String name = mbeanIntf.getName(); + Optional> op = getMap().values() + .stream() + .filter(pc -> pc.mbeanInterfaceNames().contains(name)) + .reduce((p1, p2) -> { + if (p2 != null) { + throw new IllegalArgumentException(mbeanIntf.getName() + + " can have more than one instance"); + } else { + return p1; + } + }); + + PlatformComponent singleton = op.isPresent() ? op.get() : null; + if (singleton == null) { + throw new IllegalArgumentException(mbeanIntf.getName() + + " is not a platform management interface"); + } + if (!singleton.isSingleton()) { + throw new IllegalArgumentException(mbeanIntf.getName() + + " can have more than one instance"); + } + return singleton; } } } diff --git a/jdk/src/java.management/share/classes/java/lang/management/PlatformComponent.java b/jdk/src/java.management/share/classes/java/lang/management/PlatformComponent.java deleted file mode 100644 index 0c67543eec4..00000000000 --- a/jdk/src/java.management/share/classes/java/lang/management/PlatformComponent.java +++ /dev/null @@ -1,468 +0,0 @@ -/* - * Copyright (c) 2008, 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. Oracle designates this - * particular file as subject to the "Classpath" exception as provided - * by Oracle in the LICENSE file that accompanied this code. - * - * 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. - */ - -package java.lang.management; - -import java.util.ArrayList; -import java.util.Collections; -import java.util.List; -import java.util.HashSet; -import java.util.HashMap; -import java.util.Map; -import java.util.Set; -import javax.management.MBeanServerConnection; -import javax.management.ObjectName; - -import com.sun.management.HotSpotDiagnosticMXBean; -import com.sun.management.UnixOperatingSystemMXBean; - -import sun.management.ManagementFactoryHelper; -import sun.management.Util; - -/** - * This enum class defines the list of platform components - * that provides monitoring and management support. - * Each enum represents one MXBean interface. A MXBean - * instance could implement one or more MXBean interfaces. - * - * For example, com.sun.management.GarbageCollectorMXBean - * extends java.lang.management.GarbageCollectorMXBean - * and there is one set of garbage collection MXBean instances, - * each of which implements both c.s.m. and j.l.m. interfaces. - * There are two separate enums GARBAGE_COLLECTOR - * and SUN_GARBAGE_COLLECTOR so that ManagementFactory.getPlatformMXBeans(Class) - * will return the list of MXBeans of the specified type. - * - * To add a new MXBean interface for the Java platform, - * add a new enum constant and implement the MXBeanFetcher. - */ -enum PlatformComponent { - - /** - * Class loading system of the Java virtual machine. - */ - CLASS_LOADING( - "java.lang.management.ClassLoadingMXBean", - "java.lang", "ClassLoading", defaultKeyProperties(), - true, // singleton - new MXBeanFetcher() { - public List getMXBeans() { - return Collections.singletonList(ManagementFactoryHelper.getClassLoadingMXBean()); - } - }), - - /** - * Compilation system of the Java virtual machine. - */ - COMPILATION( - "java.lang.management.CompilationMXBean", - "java.lang", "Compilation", defaultKeyProperties(), - true, // singleton - new MXBeanFetcher() { - public List getMXBeans() { - CompilationMXBean m = ManagementFactoryHelper.getCompilationMXBean(); - if (m == null) { - return Collections.emptyList(); - } else { - return Collections.singletonList(m); - } - } - }), - - /** - * Memory system of the Java virtual machine. - */ - MEMORY( - "java.lang.management.MemoryMXBean", - "java.lang", "Memory", defaultKeyProperties(), - true, // singleton - new MXBeanFetcher() { - public List getMXBeans() { - return Collections.singletonList(ManagementFactoryHelper.getMemoryMXBean()); - } - }), - - /** - * Garbage Collector in the Java virtual machine. - */ - GARBAGE_COLLECTOR( - "java.lang.management.GarbageCollectorMXBean", - "java.lang", "GarbageCollector", keyProperties("name"), - false, // zero or more instances - new MXBeanFetcher() { - public List getMXBeans() { - return ManagementFactoryHelper. - getGarbageCollectorMXBeans(); - } - }), - - /** - * Memory manager in the Java virtual machine. - */ - MEMORY_MANAGER( - "java.lang.management.MemoryManagerMXBean", - "java.lang", "MemoryManager", keyProperties("name"), - false, // zero or more instances - new MXBeanFetcher() { - public List getMXBeans() { - return ManagementFactoryHelper.getMemoryManagerMXBeans(); - } - }, - GARBAGE_COLLECTOR), - - /** - * Memory pool in the Java virtual machine. - */ - MEMORY_POOL( - "java.lang.management.MemoryPoolMXBean", - "java.lang", "MemoryPool", keyProperties("name"), - false, // zero or more instances - new MXBeanFetcher() { - public List getMXBeans() { - return ManagementFactoryHelper.getMemoryPoolMXBeans(); - } - }), - - /** - * Operating system on which the Java virtual machine is running - */ - OPERATING_SYSTEM( - "java.lang.management.OperatingSystemMXBean", - "java.lang", "OperatingSystem", defaultKeyProperties(), - true, // singleton - new MXBeanFetcher() { - public List getMXBeans() { - return Collections.singletonList(ManagementFactoryHelper.getOperatingSystemMXBean()); - } - }), - - /** - * Runtime system of the Java virtual machine. - */ - RUNTIME( - "java.lang.management.RuntimeMXBean", - "java.lang", "Runtime", defaultKeyProperties(), - true, // singleton - new MXBeanFetcher() { - public List getMXBeans() { - return Collections.singletonList(ManagementFactoryHelper.getRuntimeMXBean()); - } - }), - - /** - * Threading system of the Java virtual machine. - */ - THREADING( - "java.lang.management.ThreadMXBean", - "java.lang", "Threading", defaultKeyProperties(), - true, // singleton - new MXBeanFetcher() { - public List getMXBeans() { - return Collections.singletonList(ManagementFactoryHelper.getThreadMXBean()); - } - }), - - - /** - * Logging facility. - */ - LOGGING( - "java.lang.management.PlatformLoggingMXBean", - "java.util.logging", "Logging", defaultKeyProperties(), - true, // singleton - new MXBeanFetcher() { - public List getMXBeans() { - PlatformLoggingMXBean m = ManagementFactoryHelper.getPlatformLoggingMXBean(); - if (m == null) { - return Collections.emptyList(); - } else { - return Collections.singletonList(m); - } - } - }), - - /** - * Buffer pools. - */ - BUFFER_POOL( - "java.lang.management.BufferPoolMXBean", - "java.nio", "BufferPool", keyProperties("name"), - false, // zero or more instances - new MXBeanFetcher() { - public List getMXBeans() { - return ManagementFactoryHelper.getBufferPoolMXBeans(); - } - }), - - - // Sun Platform Extension - - /** - * Sun extension garbage collector that performs collections in cycles. - */ - SUN_GARBAGE_COLLECTOR( - "com.sun.management.GarbageCollectorMXBean", - "java.lang", "GarbageCollector", keyProperties("name"), - false, // zero or more instances - new MXBeanFetcher() { - public List getMXBeans() { - return getGcMXBeanList(com.sun.management.GarbageCollectorMXBean.class); - } - }), - - /** - * Sun extension operating system on which the Java virtual machine - * is running. - */ - SUN_OPERATING_SYSTEM( - "com.sun.management.OperatingSystemMXBean", - "java.lang", "OperatingSystem", defaultKeyProperties(), - true, // singleton - new MXBeanFetcher() { - public List getMXBeans() { - return getOSMXBeanList(com.sun.management.OperatingSystemMXBean.class); - } - }), - - /** - * Unix operating system. - */ - SUN_UNIX_OPERATING_SYSTEM( - "com.sun.management.UnixOperatingSystemMXBean", - "java.lang", "OperatingSystem", defaultKeyProperties(), - true, // singleton - new MXBeanFetcher() { - public List getMXBeans() { - return getOSMXBeanList(com.sun.management.UnixOperatingSystemMXBean.class); - } - }), - - /** - * Diagnostic support for the HotSpot Virtual Machine. - */ - HOTSPOT_DIAGNOSTIC( - "com.sun.management.HotSpotDiagnosticMXBean", - "com.sun.management", "HotSpotDiagnostic", defaultKeyProperties(), - true, // singleton - new MXBeanFetcher() { - public List getMXBeans() { - return Collections.singletonList(ManagementFactoryHelper.getDiagnosticMXBean()); - } - }); - - - /** - * A task that returns the MXBeans for a component. - */ - interface MXBeanFetcher { - public List getMXBeans(); - } - - /* - * Returns a list of the GC MXBeans of the given type. - */ - private static - List getGcMXBeanList(Class gcMXBeanIntf) { - List list = - ManagementFactoryHelper.getGarbageCollectorMXBeans(); - List result = new ArrayList<>(list.size()); - for (GarbageCollectorMXBean m : list) { - if (gcMXBeanIntf.isInstance(m)) { - result.add(gcMXBeanIntf.cast(m)); - } - } - return result; - } - - /* - * Returns the OS mxbean instance of the given type. - */ - private static - List getOSMXBeanList(Class osMXBeanIntf) { - OperatingSystemMXBean m = - ManagementFactoryHelper.getOperatingSystemMXBean(); - if (osMXBeanIntf.isInstance(m)) { - return Collections.singletonList(osMXBeanIntf.cast(m)); - } else { - return Collections.emptyList(); - } - } - - private final String mxbeanInterfaceName; - private final String domain; - private final String type; - private final Set keyProperties; - private final MXBeanFetcher fetcher; - private final PlatformComponent[] subComponents; - private final boolean singleton; - - private PlatformComponent(String intfName, - String domain, String type, - Set keyProperties, - boolean singleton, - MXBeanFetcher fetcher, - PlatformComponent... subComponents) { - this.mxbeanInterfaceName = intfName; - this.domain = domain; - this.type = type; - this.keyProperties = keyProperties; - this.singleton = singleton; - this.fetcher = fetcher; - this.subComponents = subComponents; - } - - private static Set defaultKeyProps; - private static Set defaultKeyProperties() { - if (defaultKeyProps == null) { - defaultKeyProps = Collections.singleton("type"); - } - return defaultKeyProps; - } - - private static Set keyProperties(String... keyNames) { - Set set = new HashSet<>(); - set.add("type"); - for (String s : keyNames) { - set.add(s); - } - return set; - } - - boolean isSingleton() { - return singleton; - } - - String getMXBeanInterfaceName() { - return mxbeanInterfaceName; - } - - @SuppressWarnings("unchecked") - Class getMXBeanInterface() { - try { - // Lazy loading the MXBean interface only when it is needed - return (Class) - Class.forName(mxbeanInterfaceName, false, - PlatformManagedObject.class.getClassLoader()); - } catch (ClassNotFoundException x) { - throw new AssertionError(x); - } - } - - @SuppressWarnings("unchecked") - - List getMXBeans(Class mxbeanInterface) - { - return (List) fetcher.getMXBeans(); - } - - T getSingletonMXBean(Class mxbeanInterface) - { - if (!singleton) - throw new IllegalArgumentException(mxbeanInterfaceName + - " can have zero or more than one instances"); - - List list = getMXBeans(mxbeanInterface); - assert list.size() == 1; - return list.isEmpty() ? null : list.get(0); - } - - - T getSingletonMXBean(MBeanServerConnection mbs, Class mxbeanInterface) - throws java.io.IOException - { - if (!singleton) - throw new IllegalArgumentException(mxbeanInterfaceName + - " can have zero or more than one instances"); - - // ObjectName of a singleton MXBean contains only domain and type - assert keyProperties.size() == 1; - String on = domain + ":type=" + type; - return ManagementFactory.newPlatformMXBeanProxy(mbs, - on, - mxbeanInterface); - } - - - List getMXBeans(MBeanServerConnection mbs, Class mxbeanInterface) - throws java.io.IOException - { - List result = new ArrayList<>(); - for (ObjectName on : getObjectNames(mbs)) { - result.add(ManagementFactory. - newPlatformMXBeanProxy(mbs, - on.getCanonicalName(), - mxbeanInterface) - ); - } - return result; - } - - private Set getObjectNames(MBeanServerConnection mbs) - throws java.io.IOException - { - String domainAndType = domain + ":type=" + type; - if (keyProperties.size() > 1) { - // if there are more than 1 key properties (i.e. other than "type") - domainAndType += ",*"; - } - ObjectName on = Util.newObjectName(domainAndType); - Set set = mbs.queryNames(on, null); - for (PlatformComponent pc : subComponents) { - set.addAll(pc.getObjectNames(mbs)); - } - return set; - } - - // a map from MXBean interface name to PlatformComponent - private static Map enumMap; - private static synchronized void ensureInitialized() { - if (enumMap == null) { - enumMap = new HashMap<>(); - for (PlatformComponent pc: PlatformComponent.values()) { - // Use String as the key rather than Class to avoid - // causing unnecessary class loading of management interface - enumMap.put(pc.getMXBeanInterfaceName(), pc); - } - } - } - - static boolean isPlatformMXBean(String cn) { - ensureInitialized(); - return enumMap.containsKey(cn); - } - - static - PlatformComponent getPlatformComponent(Class mxbeanInterface) - { - ensureInitialized(); - String cn = mxbeanInterface.getName(); - PlatformComponent pc = enumMap.get(cn); - if (pc != null && pc.getMXBeanInterface() == mxbeanInterface) - return pc; - return null; - } - - private static final long serialVersionUID = 6992337162326171013L; -} diff --git a/jdk/src/java.management/share/classes/sun/management/ExtendedPlatformComponent.java b/jdk/src/java.management/share/classes/sun/management/ExtendedPlatformComponent.java deleted file mode 100644 index ff33ff9b52e..00000000000 --- a/jdk/src/java.management/share/classes/sun/management/ExtendedPlatformComponent.java +++ /dev/null @@ -1,54 +0,0 @@ -/* - * Copyright (c) 2014, 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. Oracle designates this - * particular file as subject to the "Classpath" exception as provided - * by Oracle in the LICENSE file that accompanied this code. - * - * 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. - */ - -package sun.management; - -import java.util.Collections; -import java.util.List; -import java.lang.management.PlatformManagedObject; - -/** - * Class to allow for an extended set of platform MXBeans - */ -public final class ExtendedPlatformComponent { - private ExtendedPlatformComponent() {} // Don't create any instances - - /** - * Get the extended set of platform MXBeans that should be registered in the - * platform MBeanServer, or an empty list if there are no such MXBeans. - */ - public static List getMXBeans() { - return Collections.emptyList(); - } - - /** - * Returns the extended platform MXBean implementing the given - * mxbeanInterface, or null if there is no such MXBean. - */ - public static - T getMXBean(Class mxbeanInterface) { - return null; - } -} diff --git a/jdk/src/java.management/share/classes/sun/management/ManagementFactoryHelper.java b/jdk/src/java.management/share/classes/sun/management/ManagementFactoryHelper.java index be82ddae27e..c06fb71b27d 100644 --- a/jdk/src/java.management/share/classes/sun/management/ManagementFactoryHelper.java +++ b/jdk/src/java.management/share/classes/sun/management/ManagementFactoryHelper.java @@ -42,14 +42,11 @@ import java.security.PrivilegedExceptionAction; import sun.util.logging.LoggingSupport; import java.util.ArrayList; -import java.util.Collections; import java.util.HashMap; import java.util.List; import com.sun.management.DiagnosticCommandMBean; import com.sun.management.HotSpotDiagnosticMXBean; -import static java.lang.management.ManagementFactory.*; - /** * ManagementFactoryHelper provides static factory methods to create * instances of the management interface. diff --git a/jdk/src/java.management/share/classes/sun/management/spi/PlatformMBeanProvider.java b/jdk/src/java.management/share/classes/sun/management/spi/PlatformMBeanProvider.java new file mode 100644 index 00000000000..6f224c95e68 --- /dev/null +++ b/jdk/src/java.management/share/classes/sun/management/spi/PlatformMBeanProvider.java @@ -0,0 +1,233 @@ +/* + * Copyright (c) 2015, 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. Oracle designates this + * particular file as subject to the "Classpath" exception as provided + * by Oracle in the LICENSE file that accompanied this code. + * + * 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. + */ + +package sun.management.spi; + +import java.util.Collections; +import java.util.List; +import java.util.Map; +import java.util.ServiceLoader; +import java.util.Set; +import java.util.stream.Collectors; + +/** + * The PlatformMBeanProvider class defines the abstract service interface + * that the {@link java.lang.management.ManagementFactory} will invoke to find, + * load, and register Platform MBeans. + * + * ManagementFactory loads the {@linkplain ServiceLoader#loadInstalled(java.lang.Class) + * installed providers} of this service interface and each provides the + * {@linkplain PlatformComponent platform components} that defines MXBean + * or DynamicMBean to be registered in the platform MBeanServer. + * + * A {@code PlatformMBeanProvider} will implement the {@code getPlatformComponentList()} + * method to return the list of {@code PlatformComponents} it provides. + */ +public abstract class PlatformMBeanProvider { + /** + * {@code PlatformComponent} models MBeans of a management interface supported + * by the platform. + * + * If a PlatformComponent models a singleton MBean, the {@link #getObjectNamePattern() + * ObjectName pattern} must be the {@link + * javax.management.ObjectName#getCanonicalName() canonical name} of that + * singleton MBean. Otherwise, it must be an ObjectName pattern + * that can be used to query the MBeans for this + * PlatformComponent registered in a {@code MBeanServer}. + *
+ * The {@link #getObjectNamePattern() ObjectName pattern} serves as a unique + * key for identifying the instance of PlatformComponent. It is thus illegal + * for a given {@link PlatformMBeanProvider} to export several instance of + * PlatformComponent with the same + * {@link #getObjectNamePattern() ObjectName pattern} string. + *
+ * If two different provider instances export a PlatformComponent for the + * same ObjectName pattern, only the PlatformComponent instance of the first + * provider will be taken into account. + * + * @param The higher level interface for which the MBeans modeled by + * this object should be recognized. For instance, for the {@link + * java.lang.management.ManagementFactory#getOperatingSystemMXBean() + * Operating System MXBean}, this should be {@link + * java.lang.management.OperatingSystemMXBean + * java.lang.management.OperatingSystemMXBean}. + */ + public interface PlatformComponent { + /** + * Returns the names of the management interfaces implemented by the + * MBeans modeled by this {@code PlatformComponent}. + * + * @implNote + * When {@link java.lang.management.ManagementFactory#getPlatformMXBean(java.lang.Class) + * ManagementFactory.getPlatformMXBean(mxbeanInterface)} or {@link + * java.lang.management.ManagementFactory#getPlatformMXBeans(java.lang.Class) + * ManagementFactory.getPlatformMXBeans(mxbeanInterface)} are invoked, + * this PlatformComponent instance will match only if the name of the + * given {@code mxbeanInterface} is found in this list. + * + * @return the names of the management interfaces exported by the MBeans + * modeled by this object. + */ + public Set mbeanInterfaceNames(); + + /** + * A map from ObjectName string to the MBean instance this + * {@code PlatformComponent} creates. + * + * @implNote + * If {@link #shouldRegister()} is {@code true}, this method + * will be called when the {@link java.lang.management.ManagementFactory + * #getPlatformMBeanServer() Platform MBeanServer} is initialized. + * By default, this method will also be called by {@link + * #getMBeans(java.lang.Class)}, when {@link + * java.lang.management.ManagementFactory#getPlatformMXBean(java.lang.Class) + * ManagementFactory.getPlatformMXBean(mxbeanInterface)} or {@link + * java.lang.management.ManagementFactory#getPlatformMXBeans(java.lang.Class) + * ManagementFactory.getPlatformMXBeans(mxbeanInterface)} are invoked, + * and when the name of the given {@code mxbeanInterface} is contained + * in the names of management interfaces returned by {@link + * #mbeanInterfaceNames()}. + * + * @return A map with, for each MBean, the ObjectName string as key + * and the MBean as value. + */ + public Map nameToMBeanMap(); + + /** + * An ObjectName pattern uniquely identifies the MBeans + * modeled by this {@code PlatformComponent}. + * If this instance models a singleton MBean, this must be + * the {@link + * javax.management.ObjectName#getCanonicalName() canonical name} + * of that singleton MBean. + * + * @return An ObjectName pattern uniquely identifies the MBeans + * modeled by this instance. + */ + public String getObjectNamePattern(); + + /** + * Returns {@code true} if this {@code PlatformComponent} models + * a singleton MBean. By default, {@code true} is assumed. + * + * @return {@code true} if this instance models a singleton MBean. + */ + public default boolean isSingleton() { + return true; + } + + /** + * Returns {@code true} if the MBeans modeled by this {@code PlatformComponent} + * should automatically be registered in the {@link + * java.lang.management.ManagementFactory#getPlatformMBeanServer() + * Platform MBeanServer}. By default, {@code true} is assumed. + * + * @return {@code true} if the MBeans modeled by this instance should + * automatically be registered in the Platform MBeanServer. + */ + public default boolean shouldRegister() { + return true; + } + + /** + * The set of interfaces implemented by the MBeans modeled + * by this {@code PlatformComponent}. + * + * @implNote + * {@link java.lang.management.ManagementFactory#getPlatformManagementInterfaces() + * ManagementFactory.getPlatformManagementInterfaces()} calls this + * method to find the management interfaces supported by the platform. + * + * @return The set of interfaces implemented by the MBeans modeled + * by this instance + */ + public Set> mbeanInterfaces(); + + /** + * Return the list of MBeans that implement the given {@code mbeanIntf} + * modeled by this {@code PlatformComponent}. This method returns an + * empty list if no MBean implements the given {@code mbeanIntf}. + * + * @implNote This method will be called when {@link + * java.lang.management.ManagementFactory#getPlatformMXBean(java.lang.Class) + * ManagementFactory.getPlatformMXBean(mbeanIntf)} or {@link + * java.lang.management.ManagementFactory#getPlatformMXBeans(java.lang.Class) + * ManagementFactory.getPlatformMXBeans(mbeanIntf)} are invoked. + * By default it first checks whether the specified {@code mbeanIntf} + * name is contained in the returned list from the {@link #mbeanInterfaceNames()} + * method. If yes, it proceeds and calls + * {@link #mbeans().values()} and filters out all + * MBeans which are not instances of the given {@code mbeanIntf}. + * Otherwise, it returns an empty list. + * + * @param mbeanIntf A management interface. + * @return A (possibly empty) list of MBeans implementing the given + * {@code mbeanIntf}. + */ + public default List getMBeans(Class mbeanIntf) { + List list; + + if (!mbeanInterfaceNames().contains(mbeanIntf.getName())) { + list = Collections.emptyList(); + } else { + list = nameToMBeanMap().values().stream() + .filter(mbeanIntf::isInstance) + .map(mbeanIntf::cast) + .collect(Collectors.toList()); + } + return list; + } + } + + /** + * Instantiates a new PlatformMBeanProvider. + * + * @throws SecurityException if the subclass (and calling code) does not + * have {@code RuntimePermission("sun.management.spi.PlatformMBeanProvider", "subclass")} + */ + protected PlatformMBeanProvider () { + this(checkSubclassPermission()); + } + + private PlatformMBeanProvider(Void unused) { + } + + /** + * Returns a list of PlatformComponent instances describing the Platform + * MBeans provided by this provider. + * + * @return a list of PlatformComponent instances describing the Platform + * MBeans provided by this provider. + */ + public abstract List> getPlatformComponentList(); + + private static Void checkSubclassPermission() { + SecurityManager sm = System.getSecurityManager(); + if (sm != null) { + sm.checkPermission(new RuntimePermission(PlatformMBeanProvider.class.getName(), "subclass")); + } + return null; + } +} From 61473e39ce54827aa3fc5b2c1bafe56d8983fc9f Mon Sep 17 00:00:00 2001 From: Aleksei Efimov Date: Thu, 5 Feb 2015 14:48:57 +0300 Subject: [PATCH 46/58] 8072042: (tz) Support tzdata2015a Reviewed-by: coffeys, okutsu --- jdk/make/data/tzdata/VERSION | 2 +- jdk/make/data/tzdata/antarctica | 9 +++--- jdk/make/data/tzdata/asia | 28 +++++------------- jdk/make/data/tzdata/backward | 2 +- jdk/make/data/tzdata/europe | 29 +++++++++---------- jdk/make/data/tzdata/leapseconds | 4 +++ jdk/make/data/tzdata/northamerica | 27 ++++++++++++++--- jdk/make/data/tzdata/southamerica | 17 +++++++---- jdk/make/data/tzdata/zone.tab | 2 +- .../sun/util/resources/TimeZoneNames.java | 4 +-- .../util/resources/de/TimeZoneNames_de.java | 4 +-- .../util/resources/es/TimeZoneNames_es.java | 4 +-- .../util/resources/fr/TimeZoneNames_fr.java | 4 +-- .../util/resources/it/TimeZoneNames_it.java | 4 +-- .../util/resources/ja/TimeZoneNames_ja.java | 4 +-- .../util/resources/ko/TimeZoneNames_ko.java | 4 +-- .../resources/pt/TimeZoneNames_pt_BR.java | 4 +-- .../util/resources/sv/TimeZoneNames_sv.java | 4 +-- .../resources/zh/TimeZoneNames_zh_CN.java | 4 +-- .../resources/zh/TimeZoneNames_zh_TW.java | 4 +-- jdk/test/sun/util/calendar/zi/tzdata/VERSION | 2 +- .../sun/util/calendar/zi/tzdata/antarctica | 9 +++--- jdk/test/sun/util/calendar/zi/tzdata/asia | 28 +++++------------- jdk/test/sun/util/calendar/zi/tzdata/backward | 2 +- jdk/test/sun/util/calendar/zi/tzdata/europe | 29 +++++++++---------- .../sun/util/calendar/zi/tzdata/leapseconds | 4 +++ .../sun/util/calendar/zi/tzdata/northamerica | 27 ++++++++++++++--- .../sun/util/calendar/zi/tzdata/southamerica | 17 +++++++---- jdk/test/sun/util/calendar/zi/tzdata/zone.tab | 2 +- 29 files changed, 158 insertions(+), 126 deletions(-) diff --git a/jdk/make/data/tzdata/VERSION b/jdk/make/data/tzdata/VERSION index 9987fde6dcb..034114ae6ca 100644 --- a/jdk/make/data/tzdata/VERSION +++ b/jdk/make/data/tzdata/VERSION @@ -21,4 +21,4 @@ # or visit www.oracle.com if you need additional information or have any # questions. # -tzdata2014j +tzdata2015a diff --git a/jdk/make/data/tzdata/antarctica b/jdk/make/data/tzdata/antarctica index 0cdac270861..1f3e4347b21 100644 --- a/jdk/make/data/tzdata/antarctica +++ b/jdk/make/data/tzdata/antarctica @@ -70,8 +70,8 @@ Rule ChileAQ 2009 only - Mar Sun>=9 3:00u 0 - Rule ChileAQ 2010 only - Apr Sun>=1 3:00u 0 - Rule ChileAQ 2011 only - May Sun>=2 3:00u 0 - Rule ChileAQ 2011 only - Aug Sun>=16 4:00u 1:00 S -Rule ChileAQ 2012 max - Apr Sun>=23 3:00u 0 - -Rule ChileAQ 2012 max - Sep Sun>=2 4:00u 1:00 S +Rule ChileAQ 2012 2015 - Apr Sun>=23 3:00u 0 - +Rule ChileAQ 2012 2014 - Sep Sun>=2 4:00u 1:00 S # Argentina - year-round bases # Belgrano II, Confin Coast, -770227-0343737, since 1972-02-05 @@ -377,9 +377,10 @@ Zone Antarctica/Rothera 0 - zzz 1976 Dec 1 # # Zone NAME GMTOFF RULES FORMAT [UNTIL] Zone Antarctica/Palmer 0 - zzz 1965 - -4:00 ArgAQ AR%sT 1969 Oct 5 + -4:00 ArgAQ AR%sT 1969 Oct 5 -3:00 ArgAQ AR%sT 1982 May - -4:00 ChileAQ CL%sT + -4:00 ChileAQ CL%sT 2015 Apr 26 3:00u + -3:00 - CLT # # # McMurdo Station, Ross Island, since 1955-12 diff --git a/jdk/make/data/tzdata/asia b/jdk/make/data/tzdata/asia index 960cab5062e..bff837c48d8 100644 --- a/jdk/make/data/tzdata/asia +++ b/jdk/make/data/tzdata/asia @@ -168,10 +168,7 @@ Zone Asia/Baku 3:19:24 - LMT 1924 May 2 4:00 Azer AZ%sT # Bahrain -# Zone NAME GMTOFF RULES FORMAT [UNTIL] -Zone Asia/Bahrain 3:22:20 - LMT 1920 # Manamah - 4:00 - GST 1972 Jun - 3:00 - AST +# See Asia/Qatar. # Bangladesh # From Alexander Krivenyshev (2009-05-13): @@ -1754,9 +1751,7 @@ Zone Asia/Pyongyang 8:23:00 - LMT 1908 Apr 1 ############################################################################### # Kuwait -# Zone NAME GMTOFF RULES FORMAT [UNTIL] -Zone Asia/Kuwait 3:11:56 - LMT 1950 - 3:00 - AST +# See Asia/Riyadh. # Laos # See Asia/Bangkok. @@ -1977,12 +1972,7 @@ Zone Asia/Kathmandu 5:41:16 - LMT 1920 5:45 - NPT # Nepal Time # Oman - -# Milne says 3:54:24 was the meridian of the Muscat Tidal Observatory. - -# Zone NAME GMTOFF RULES FORMAT [UNTIL] -Zone Asia/Muscat 3:54:24 - LMT 1920 - 4:00 - GST +# See Asia/Dubai. # Pakistan @@ -2476,6 +2466,7 @@ Zone Asia/Manila -15:56:00 - LMT 1844 Dec 31 Zone Asia/Qatar 3:26:08 - LMT 1920 # Al Dawhah / Doha 4:00 - GST 1972 Jun 3:00 - AST +Link Asia/Qatar Asia/Bahrain # Saudi Arabia # @@ -2502,6 +2493,8 @@ Zone Asia/Qatar 3:26:08 - LMT 1920 # Al Dawhah / Doha # Zone NAME GMTOFF RULES FORMAT [UNTIL] Zone Asia/Riyadh 3:06:52 - LMT 1947 Mar 14 3:00 - AST +Link Asia/Riyadh Asia/Aden # Yemen +Link Asia/Riyadh Asia/Kuwait # Singapore # taken from Mok Ly Yng (2003-10-30) @@ -2790,6 +2783,7 @@ Zone Asia/Ashgabat 3:53:32 - LMT 1924 May 2 # or Ashkhabad # Zone NAME GMTOFF RULES FORMAT [UNTIL] Zone Asia/Dubai 3:41:12 - LMT 1920 4:00 - GST +Link Asia/Dubai Asia/Muscat # Oman # Uzbekistan # Byalokoz 1919 says Uzbekistan was 4:27:53. @@ -2874,10 +2868,4 @@ Zone Asia/Ho_Chi_Minh 7:06:40 - LMT 1906 Jul 1 7:00 - ICT # Yemen - -# Milne says 2:59:54 was the meridian of the saluting battery at Aden, -# and that Yemen was at 1:55:56, the meridian of the Hagia Sophia. - -# Zone NAME GMTOFF RULES FORMAT [UNTIL] -Zone Asia/Aden 2:59:54 - LMT 1950 - 3:00 - AST +# See Asia/Riyadh. diff --git a/jdk/make/data/tzdata/backward b/jdk/make/data/tzdata/backward index ba012f45733..95266a6f16c 100644 --- a/jdk/make/data/tzdata/backward +++ b/jdk/make/data/tzdata/backward @@ -28,7 +28,7 @@ # and their old names. Many names changed in late 1993. # Link TARGET LINK-NAME -Link Africa/Asmara Africa/Asmera +Link Africa/Nairobi Africa/Asmera Link Africa/Abidjan Africa/Timbuktu Link America/Argentina/Catamarca America/Argentina/ComodRivadavia Link America/Adak America/Atka diff --git a/jdk/make/data/tzdata/europe b/jdk/make/data/tzdata/europe index 2ed6ad36b5d..89790f06c17 100644 --- a/jdk/make/data/tzdata/europe +++ b/jdk/make/data/tzdata/europe @@ -1430,35 +1430,32 @@ Zone Europe/Budapest 1:16:20 - LMT 1890 Oct # might be a reference to the Julian calendar as opposed to Gregorian, or it # might mean something else (???). # -# From Paul Eggert (2006-03-22): -# The Iceland Almanak, Shanks & Pottenger, and Whitman disagree on many points. -# We go with the Almanak, except for one claim from Shanks & Pottenger, namely -# that Reykavik was 21W57 from 1837 to 1908, local mean time before that. +# From Paul Eggert (2014-11-22): +# The information below is taken from the 1988 Almanak; see +# http://www.almanak.hi.is/klukkan.html # # Rule NAME FROM TO TYPE IN ON AT SAVE LETTER/S -Rule Iceland 1917 1918 - Feb 19 23:00 1:00 S +Rule Iceland 1917 1919 - Feb 19 23:00 1:00 S Rule Iceland 1917 only - Oct 21 1:00 0 - -Rule Iceland 1918 only - Nov 16 1:00 0 - +Rule Iceland 1918 1919 - Nov 16 1:00 0 - +Rule Iceland 1921 only - Mar 19 23:00 1:00 S +Rule Iceland 1921 only - Jun 23 1:00 0 - Rule Iceland 1939 only - Apr 29 23:00 1:00 S -Rule Iceland 1939 only - Nov 29 2:00 0 - +Rule Iceland 1939 only - Oct 29 2:00 0 - Rule Iceland 1940 only - Feb 25 2:00 1:00 S -Rule Iceland 1940 only - Nov 3 2:00 0 - -Rule Iceland 1941 only - Mar 2 1:00s 1:00 S -Rule Iceland 1941 only - Nov 2 1:00s 0 - -Rule Iceland 1942 only - Mar 8 1:00s 1:00 S -Rule Iceland 1942 only - Oct 25 1:00s 0 - +Rule Iceland 1940 1941 - Nov Sun>=2 1:00s 0 - +Rule Iceland 1941 1942 - Mar Sun>=2 1:00s 1:00 S # 1943-1946 - first Sunday in March until first Sunday in winter Rule Iceland 1943 1946 - Mar Sun>=1 1:00s 1:00 S -Rule Iceland 1943 1948 - Oct Sun>=22 1:00s 0 - +Rule Iceland 1942 1948 - Oct Sun>=22 1:00s 0 - # 1947-1967 - first Sunday in April until first Sunday in winter Rule Iceland 1947 1967 - Apr Sun>=1 1:00s 1:00 S -# 1949 Oct transition delayed by 1 week +# 1949 and 1967 Oct transitions delayed by 1 week Rule Iceland 1949 only - Oct 30 1:00s 0 - Rule Iceland 1950 1966 - Oct Sun>=22 1:00s 0 - Rule Iceland 1967 only - Oct 29 1:00s 0 - # Zone NAME GMTOFF RULES FORMAT [UNTIL] -Zone Atlantic/Reykjavik -1:27:24 - LMT 1837 - -1:27:48 - RMT 1908 # Reykjavik Mean Time? +Zone Atlantic/Reykjavik -1:28 - LMT 1908 -1:00 Iceland IS%sT 1968 Apr 7 1:00s 0:00 - GMT diff --git a/jdk/make/data/tzdata/leapseconds b/jdk/make/data/tzdata/leapseconds index 7612f2bc9b7..9b0a2278433 100644 --- a/jdk/make/data/tzdata/leapseconds +++ b/jdk/make/data/tzdata/leapseconds @@ -77,3 +77,7 @@ Leap 1998 Dec 31 23:59:60 + S Leap 2005 Dec 31 23:59:60 + S Leap 2008 Dec 31 23:59:60 + S Leap 2012 Jun 30 23:59:60 + S +Leap 2015 Jun 30 23:59:60 + S + +# Updated through IERS Bulletin C49 +# File expires on: 28 December 2015 diff --git a/jdk/make/data/tzdata/northamerica b/jdk/make/data/tzdata/northamerica index 86c9503a4eb..5943cfeec4b 100644 --- a/jdk/make/data/tzdata/northamerica +++ b/jdk/make/data/tzdata/northamerica @@ -147,7 +147,7 @@ Rule US 1918 1919 - Mar lastSun 2:00 1:00 D Rule US 1918 1919 - Oct lastSun 2:00 0 S Rule US 1942 only - Feb 9 2:00 1:00 W # War Rule US 1945 only - Aug 14 23:00u 1:00 P # Peace -Rule US 1945 only - Sep 30 2:00 0 S +Rule US 1945 only - Sep lastSun 2:00 0 S Rule US 1967 2006 - Oct lastSun 2:00 0 S Rule US 1967 1973 - Apr lastSun 2:00 1:00 D Rule US 1974 only - Jan 6 2:00 1:00 D @@ -2147,11 +2147,11 @@ Zone America/Dawson -9:17:40 - LMT 1900 Aug 20 # Mexico -# From Paul Eggert (2001-03-05): +# From Paul Eggert (2014-12-07): # The Investigation and Analysis Service of the # Mexican Library of Congress (MLoC) has published a # history of Mexican local time (in Spanish) -# http://www.cddhcu.gob.mx/bibliot/publica/inveyana/polisoc/horver/ +# http://www.diputados.gob.mx/bibliot/publica/inveyana/polisoc/horver/index.htm # # Here are the discrepancies between Shanks & Pottenger (S&P) and the MLoC. # (In all cases we go with the MLoC.) @@ -2320,6 +2320,24 @@ Zone America/Dawson -9:17:40 - LMT 1900 Aug 20 # efecto desde las dos horas del segundo domingo de marzo y concluirá a # las dos horas del primer domingo de noviembre. +# From Steffen Thorsen (2014-12-08), translated by Gwillim Law: +# The Mexican state of Quintana Roo will likely change to EST in 2015. +# +# http://www.unioncancun.mx/articulo/2014/12/04/medio-ambiente/congreso-aprueba-una-hora-mas-de-sol-en-qroo +# "With this change, the time conflict that has existed between the municipios +# of Quintana Roo and the municipio of Felipe Carrillo Puerto may come to an +# end. The latter declared itself in rebellion 15 years ago when a time change +# was initiated in Mexico, and since then it has refused to change its time +# zone along with the rest of the country." +# +# From Steffen Thorsen (2015-01-14), translated by Gwillim Law: +# http://sipse.com/novedades/confirman-aplicacion-de-nueva-zona-horaria-para-quintana-roo-132331.html +# "...the new time zone will come into effect at two o'clock on the first Sunday +# of February, when we will have to advance the clock one hour from its current +# time..." +# +# Also, the new zone will not use DST. + # Rule NAME FROM TO TYPE IN ON AT SAVE LETTER/S Rule Mexico 1939 only - Feb 5 0:00 1:00 D Rule Mexico 1939 only - Jun 25 0:00 0 S @@ -2340,7 +2358,8 @@ Rule Mexico 2002 max - Oct lastSun 2:00 0 S Zone America/Cancun -5:47:04 - LMT 1922 Jan 1 0:12:56 -6:00 - CST 1981 Dec 23 -5:00 Mexico E%sT 1998 Aug 2 2:00 - -6:00 Mexico C%sT + -6:00 Mexico C%sT 2015 Feb 1 2:00 + -5:00 - EST # Campeche, Yucatán; represented by Mérida Zone America/Merida -5:58:28 - LMT 1922 Jan 1 0:01:32 -6:00 - CST 1981 Dec 23 diff --git a/jdk/make/data/tzdata/southamerica b/jdk/make/data/tzdata/southamerica index 0b70dea5616..02cf12113a0 100644 --- a/jdk/make/data/tzdata/southamerica +++ b/jdk/make/data/tzdata/southamerica @@ -1229,6 +1229,11 @@ Zone America/Rio_Branco -4:31:12 - LMT 1914 # DST Start: first Saturday of September 2014 (Sun 07 Sep 2014 04:00 UTC) # http://www.diariooficial.interior.gob.cl//media/2014/02/19/do-20140219.pdf +# From Juan Correa (2015-01-28): +# ... today the Ministry of Energy announced that Chile will drop DST, will keep +# "summer time" (UTC -3 / UTC -5) all year round.... +# http://www.minenergia.cl/ministerio/noticias/generales/ministerio-de-energia-anuncia.html + # NOTE: ChileAQ rules for Antarctic bases are stored separately in the # 'antarctica' file. @@ -1270,8 +1275,8 @@ Rule Chile 2009 only - Mar Sun>=9 3:00u 0 - Rule Chile 2010 only - Apr Sun>=1 3:00u 0 - Rule Chile 2011 only - May Sun>=2 3:00u 0 - Rule Chile 2011 only - Aug Sun>=16 4:00u 1:00 S -Rule Chile 2012 max - Apr Sun>=23 3:00u 0 - -Rule Chile 2012 max - Sep Sun>=2 4:00u 1:00 S +Rule Chile 2012 2015 - Apr Sun>=23 3:00u 0 - +Rule Chile 2012 2014 - Sep Sun>=2 4:00u 1:00 S # IATA SSIM anomalies: (1992-02) says 1992-03-14; # (1996-09) says 1998-03-08. Ignore these. # Zone NAME GMTOFF RULES FORMAT [UNTIL] @@ -1282,11 +1287,13 @@ Zone America/Santiago -4:42:46 - LMT 1890 -4:00 - CLT 1919 Jul 1 # Chile Time -4:42:46 - SMT 1927 Sep 1 # Santiago Mean Time -5:00 Chile CL%sT 1947 May 22 # Chile Time - -4:00 Chile CL%sT + -4:00 Chile CL%sT 2015 Apr 26 3:00u + -3:00 - CLT Zone Pacific/Easter -7:17:44 - LMT 1890 -7:17:28 - EMT 1932 Sep # Easter Mean Time - -7:00 Chile EAS%sT 1982 Mar 13 21:00 # Easter Time - -6:00 Chile EAS%sT + -7:00 Chile EAS%sT 1982 Mar 13 3:00u # Easter Time + -6:00 Chile EAS%sT 2015 Apr 26 3:00u + -5:00 - EAST # # Salas y Gómez Island is uninhabited. # Other Chilean locations, including Juan Fernández Is, Desventuradas Is, diff --git a/jdk/make/data/tzdata/zone.tab b/jdk/make/data/tzdata/zone.tab index 0ef9ba869ea..ffb6469676e 100644 --- a/jdk/make/data/tzdata/zone.tab +++ b/jdk/make/data/tzdata/zone.tab @@ -297,7 +297,7 @@ MU -2010+05730 Indian/Mauritius MV +0410+07330 Indian/Maldives MW -1547+03500 Africa/Blantyre MX +1924-09909 America/Mexico_City Central Time - most locations -MX +2105-08646 America/Cancun Central Time - Quintana Roo +MX +2105-08646 America/Cancun Eastern Standard Time - Quintana Roo MX +2058-08937 America/Merida Central Time - Campeche, Yucatan MX +2540-10019 America/Monterrey Mexican Central Time - Coahuila, Durango, Nuevo Leon, Tamaulipas away from US border MX +2550-09730 America/Matamoros US Central Time - Coahuila, Durango, Nuevo Leon, Tamaulipas near US border diff --git a/jdk/src/java.base/share/classes/sun/util/resources/TimeZoneNames.java b/jdk/src/java.base/share/classes/sun/util/resources/TimeZoneNames.java index 679a5c10033..75efc60e0db 100644 --- a/jdk/src/java.base/share/classes/sun/util/resources/TimeZoneNames.java +++ b/jdk/src/java.base/share/classes/sun/util/resources/TimeZoneNames.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1996, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1996, 2015, 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 @@ -408,7 +408,7 @@ public final class TimeZoneNames extends TimeZoneNamesBundle { {"America/Buenos_Aires", AGT}, {"America/Cambridge_Bay", MST}, {"America/Campo_Grande", AMT}, - {"America/Cancun", CST}, + {"America/Cancun", EST}, {"America/Caracas", new String[] {"Venezuela Time", "VET", "Venezuela Summer Time", "VEST", "Venezuela Time", "VET"}}, diff --git a/jdk/src/jdk.localedata/share/classes/sun/util/resources/de/TimeZoneNames_de.java b/jdk/src/jdk.localedata/share/classes/sun/util/resources/de/TimeZoneNames_de.java index 83ba779778b..7dd97ebd058 100644 --- a/jdk/src/jdk.localedata/share/classes/sun/util/resources/de/TimeZoneNames_de.java +++ b/jdk/src/jdk.localedata/share/classes/sun/util/resources/de/TimeZoneNames_de.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -409,7 +409,7 @@ public final class TimeZoneNames_de extends TimeZoneNamesBundle { {"America/Buenos_Aires", AGT}, {"America/Cambridge_Bay", MST}, {"America/Campo_Grande", AMT}, - {"America/Cancun", CST}, + {"America/Cancun", EST}, {"America/Caracas", new String[] {"Venezuelanische Zeit", "VET", "Venezuelanische Sommerzeit", "VEST", "Venezuelanische Zeit", "VET"}}, diff --git a/jdk/src/jdk.localedata/share/classes/sun/util/resources/es/TimeZoneNames_es.java b/jdk/src/jdk.localedata/share/classes/sun/util/resources/es/TimeZoneNames_es.java index 2b472669498..99f48b2d902 100644 --- a/jdk/src/jdk.localedata/share/classes/sun/util/resources/es/TimeZoneNames_es.java +++ b/jdk/src/jdk.localedata/share/classes/sun/util/resources/es/TimeZoneNames_es.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -409,7 +409,7 @@ public final class TimeZoneNames_es extends TimeZoneNamesBundle { {"America/Buenos_Aires", AGT}, {"America/Cambridge_Bay", MST}, {"America/Campo_Grande", AMT}, - {"America/Cancun", CST}, + {"America/Cancun", EST}, {"America/Caracas", new String[] {"Hora de Venezuela", "VET", "Hora de verano de Venezuela", "VEST", "Hora de Venezuela", "VET"}}, diff --git a/jdk/src/jdk.localedata/share/classes/sun/util/resources/fr/TimeZoneNames_fr.java b/jdk/src/jdk.localedata/share/classes/sun/util/resources/fr/TimeZoneNames_fr.java index 41aefbdad90..9a89c8508df 100644 --- a/jdk/src/jdk.localedata/share/classes/sun/util/resources/fr/TimeZoneNames_fr.java +++ b/jdk/src/jdk.localedata/share/classes/sun/util/resources/fr/TimeZoneNames_fr.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -409,7 +409,7 @@ public final class TimeZoneNames_fr extends TimeZoneNamesBundle { {"America/Buenos_Aires", AGT}, {"America/Cambridge_Bay", MST}, {"America/Campo_Grande", AMT}, - {"America/Cancun", CST}, + {"America/Cancun", EST}, {"America/Caracas", new String[] {"Heure du Venezuela", "VET", "Heure d'\u00e9t\u00e9 du Venezuela", "VEST", "Heure du Venezuela", "VET"}}, diff --git a/jdk/src/jdk.localedata/share/classes/sun/util/resources/it/TimeZoneNames_it.java b/jdk/src/jdk.localedata/share/classes/sun/util/resources/it/TimeZoneNames_it.java index 8f350fbf337..b3edb6e4236 100644 --- a/jdk/src/jdk.localedata/share/classes/sun/util/resources/it/TimeZoneNames_it.java +++ b/jdk/src/jdk.localedata/share/classes/sun/util/resources/it/TimeZoneNames_it.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -409,7 +409,7 @@ public final class TimeZoneNames_it extends TimeZoneNamesBundle { {"America/Buenos_Aires", AGT}, {"America/Cambridge_Bay", MST}, {"America/Campo_Grande", AMT}, - {"America/Cancun", CST}, + {"America/Cancun", EST}, {"America/Caracas", new String[] {"Ora del Venezuela", "VET", "Ora estiva del Venezuela", "VEST", "Ora del Venezuela", "VET"}}, diff --git a/jdk/src/jdk.localedata/share/classes/sun/util/resources/ja/TimeZoneNames_ja.java b/jdk/src/jdk.localedata/share/classes/sun/util/resources/ja/TimeZoneNames_ja.java index 5ae2aa07994..df5d3acbba2 100644 --- a/jdk/src/jdk.localedata/share/classes/sun/util/resources/ja/TimeZoneNames_ja.java +++ b/jdk/src/jdk.localedata/share/classes/sun/util/resources/ja/TimeZoneNames_ja.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -409,7 +409,7 @@ public final class TimeZoneNames_ja extends TimeZoneNamesBundle { {"America/Buenos_Aires", AGT}, {"America/Cambridge_Bay", MST}, {"America/Campo_Grande", AMT}, - {"America/Cancun", CST}, + {"America/Cancun", EST}, {"America/Caracas", new String[] {"\u30d9\u30cd\u30ba\u30a8\u30e9\u6642\u9593", "VET", "\u30d9\u30cd\u30ba\u30a8\u30e9\u590f\u6642\u9593", "VEST", "\u30D9\u30CD\u30BA\u30A8\u30E9\u6642\u9593", "VET"}}, diff --git a/jdk/src/jdk.localedata/share/classes/sun/util/resources/ko/TimeZoneNames_ko.java b/jdk/src/jdk.localedata/share/classes/sun/util/resources/ko/TimeZoneNames_ko.java index a35768503d1..2ef152babc1 100644 --- a/jdk/src/jdk.localedata/share/classes/sun/util/resources/ko/TimeZoneNames_ko.java +++ b/jdk/src/jdk.localedata/share/classes/sun/util/resources/ko/TimeZoneNames_ko.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -409,7 +409,7 @@ public final class TimeZoneNames_ko extends TimeZoneNamesBundle { {"America/Buenos_Aires", AGT}, {"America/Cambridge_Bay", MST}, {"America/Campo_Grande", AMT}, - {"America/Cancun", CST}, + {"America/Cancun", EST}, {"America/Caracas", new String[] {"\ubca0\ub124\uc218\uc5d8\ub77c \uc2dc\uac04", "VET", "\ubca0\ub124\uc218\uc5d8\ub77c \uc77c\uad11\uc808\uc57d\uc2dc\uac04", "VEST", "\uBCA0\uB124\uC218\uC5D8\uB77C \uD45C\uC900\uC2DC", "VET"}}, diff --git a/jdk/src/jdk.localedata/share/classes/sun/util/resources/pt/TimeZoneNames_pt_BR.java b/jdk/src/jdk.localedata/share/classes/sun/util/resources/pt/TimeZoneNames_pt_BR.java index 942081c5315..9ff7e574c84 100644 --- a/jdk/src/jdk.localedata/share/classes/sun/util/resources/pt/TimeZoneNames_pt_BR.java +++ b/jdk/src/jdk.localedata/share/classes/sun/util/resources/pt/TimeZoneNames_pt_BR.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -409,7 +409,7 @@ public final class TimeZoneNames_pt_BR extends TimeZoneNamesBundle { {"America/Buenos_Aires", AGT}, {"America/Cambridge_Bay", MST}, {"America/Campo_Grande", AMT}, - {"America/Cancun", CST}, + {"America/Cancun", EST}, {"America/Caracas", new String[] {"Fuso hor\u00e1rio da Venezuela", "VET", "Fuso hor\u00e1rio de ver\u00e3o da Venezuela", "VEST", "Hor\u00E1rio da Venezuela", "VET"}}, diff --git a/jdk/src/jdk.localedata/share/classes/sun/util/resources/sv/TimeZoneNames_sv.java b/jdk/src/jdk.localedata/share/classes/sun/util/resources/sv/TimeZoneNames_sv.java index bd77d58b881..916c2c5d981 100644 --- a/jdk/src/jdk.localedata/share/classes/sun/util/resources/sv/TimeZoneNames_sv.java +++ b/jdk/src/jdk.localedata/share/classes/sun/util/resources/sv/TimeZoneNames_sv.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -409,7 +409,7 @@ public final class TimeZoneNames_sv extends TimeZoneNamesBundle { {"America/Buenos_Aires", AGT}, {"America/Cambridge_Bay", MST}, {"America/Campo_Grande", AMT}, - {"America/Cancun", CST}, + {"America/Cancun", EST}, {"America/Caracas", new String[] {"Venezuela, normaltid", "VET", "Venezuela, sommartid", "VEST", "Venezuelansk tid", "VET"}}, diff --git a/jdk/src/jdk.localedata/share/classes/sun/util/resources/zh/TimeZoneNames_zh_CN.java b/jdk/src/jdk.localedata/share/classes/sun/util/resources/zh/TimeZoneNames_zh_CN.java index c0590e574b1..5cb70c85eb9 100644 --- a/jdk/src/jdk.localedata/share/classes/sun/util/resources/zh/TimeZoneNames_zh_CN.java +++ b/jdk/src/jdk.localedata/share/classes/sun/util/resources/zh/TimeZoneNames_zh_CN.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -409,7 +409,7 @@ public final class TimeZoneNames_zh_CN extends TimeZoneNamesBundle { {"America/Buenos_Aires", AGT}, {"America/Cambridge_Bay", MST}, {"America/Campo_Grande", AMT}, - {"America/Cancun", CST}, + {"America/Cancun", EST}, {"America/Caracas", new String[] {"\u59d4\u5185\u745e\u62c9\u65f6\u95f4", "VET", "\u59d4\u5185\u745e\u62c9\u590f\u4ee4\u65f6", "VEST", "\u59D4\u5185\u745E\u62C9\u65F6\u95F4", "VET"}}, diff --git a/jdk/src/jdk.localedata/share/classes/sun/util/resources/zh/TimeZoneNames_zh_TW.java b/jdk/src/jdk.localedata/share/classes/sun/util/resources/zh/TimeZoneNames_zh_TW.java index 19c332dea10..0074f49da8b 100644 --- a/jdk/src/jdk.localedata/share/classes/sun/util/resources/zh/TimeZoneNames_zh_TW.java +++ b/jdk/src/jdk.localedata/share/classes/sun/util/resources/zh/TimeZoneNames_zh_TW.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 1997, 2014, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 1997, 2015, 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 @@ -409,7 +409,7 @@ public final class TimeZoneNames_zh_TW extends TimeZoneNamesBundle { {"America/Buenos_Aires", AGT}, {"America/Cambridge_Bay", MST}, {"America/Campo_Grande", AMT}, - {"America/Cancun", CST}, + {"America/Cancun", EST}, {"America/Caracas", new String[] {"\u59d4\u5167\u745e\u62c9\u6642\u9593", "VET", "\u59d4\u5167\u745e\u62c9\u590f\u4ee4\u6642\u9593", "VEST", "\u59D4\u5167\u745E\u62C9\u6642\u9593", "VET"}}, diff --git a/jdk/test/sun/util/calendar/zi/tzdata/VERSION b/jdk/test/sun/util/calendar/zi/tzdata/VERSION index 9987fde6dcb..034114ae6ca 100644 --- a/jdk/test/sun/util/calendar/zi/tzdata/VERSION +++ b/jdk/test/sun/util/calendar/zi/tzdata/VERSION @@ -21,4 +21,4 @@ # or visit www.oracle.com if you need additional information or have any # questions. # -tzdata2014j +tzdata2015a diff --git a/jdk/test/sun/util/calendar/zi/tzdata/antarctica b/jdk/test/sun/util/calendar/zi/tzdata/antarctica index 0cdac270861..1f3e4347b21 100644 --- a/jdk/test/sun/util/calendar/zi/tzdata/antarctica +++ b/jdk/test/sun/util/calendar/zi/tzdata/antarctica @@ -70,8 +70,8 @@ Rule ChileAQ 2009 only - Mar Sun>=9 3:00u 0 - Rule ChileAQ 2010 only - Apr Sun>=1 3:00u 0 - Rule ChileAQ 2011 only - May Sun>=2 3:00u 0 - Rule ChileAQ 2011 only - Aug Sun>=16 4:00u 1:00 S -Rule ChileAQ 2012 max - Apr Sun>=23 3:00u 0 - -Rule ChileAQ 2012 max - Sep Sun>=2 4:00u 1:00 S +Rule ChileAQ 2012 2015 - Apr Sun>=23 3:00u 0 - +Rule ChileAQ 2012 2014 - Sep Sun>=2 4:00u 1:00 S # Argentina - year-round bases # Belgrano II, Confin Coast, -770227-0343737, since 1972-02-05 @@ -377,9 +377,10 @@ Zone Antarctica/Rothera 0 - zzz 1976 Dec 1 # # Zone NAME GMTOFF RULES FORMAT [UNTIL] Zone Antarctica/Palmer 0 - zzz 1965 - -4:00 ArgAQ AR%sT 1969 Oct 5 + -4:00 ArgAQ AR%sT 1969 Oct 5 -3:00 ArgAQ AR%sT 1982 May - -4:00 ChileAQ CL%sT + -4:00 ChileAQ CL%sT 2015 Apr 26 3:00u + -3:00 - CLT # # # McMurdo Station, Ross Island, since 1955-12 diff --git a/jdk/test/sun/util/calendar/zi/tzdata/asia b/jdk/test/sun/util/calendar/zi/tzdata/asia index 960cab5062e..bff837c48d8 100644 --- a/jdk/test/sun/util/calendar/zi/tzdata/asia +++ b/jdk/test/sun/util/calendar/zi/tzdata/asia @@ -168,10 +168,7 @@ Zone Asia/Baku 3:19:24 - LMT 1924 May 2 4:00 Azer AZ%sT # Bahrain -# Zone NAME GMTOFF RULES FORMAT [UNTIL] -Zone Asia/Bahrain 3:22:20 - LMT 1920 # Manamah - 4:00 - GST 1972 Jun - 3:00 - AST +# See Asia/Qatar. # Bangladesh # From Alexander Krivenyshev (2009-05-13): @@ -1754,9 +1751,7 @@ Zone Asia/Pyongyang 8:23:00 - LMT 1908 Apr 1 ############################################################################### # Kuwait -# Zone NAME GMTOFF RULES FORMAT [UNTIL] -Zone Asia/Kuwait 3:11:56 - LMT 1950 - 3:00 - AST +# See Asia/Riyadh. # Laos # See Asia/Bangkok. @@ -1977,12 +1972,7 @@ Zone Asia/Kathmandu 5:41:16 - LMT 1920 5:45 - NPT # Nepal Time # Oman - -# Milne says 3:54:24 was the meridian of the Muscat Tidal Observatory. - -# Zone NAME GMTOFF RULES FORMAT [UNTIL] -Zone Asia/Muscat 3:54:24 - LMT 1920 - 4:00 - GST +# See Asia/Dubai. # Pakistan @@ -2476,6 +2466,7 @@ Zone Asia/Manila -15:56:00 - LMT 1844 Dec 31 Zone Asia/Qatar 3:26:08 - LMT 1920 # Al Dawhah / Doha 4:00 - GST 1972 Jun 3:00 - AST +Link Asia/Qatar Asia/Bahrain # Saudi Arabia # @@ -2502,6 +2493,8 @@ Zone Asia/Qatar 3:26:08 - LMT 1920 # Al Dawhah / Doha # Zone NAME GMTOFF RULES FORMAT [UNTIL] Zone Asia/Riyadh 3:06:52 - LMT 1947 Mar 14 3:00 - AST +Link Asia/Riyadh Asia/Aden # Yemen +Link Asia/Riyadh Asia/Kuwait # Singapore # taken from Mok Ly Yng (2003-10-30) @@ -2790,6 +2783,7 @@ Zone Asia/Ashgabat 3:53:32 - LMT 1924 May 2 # or Ashkhabad # Zone NAME GMTOFF RULES FORMAT [UNTIL] Zone Asia/Dubai 3:41:12 - LMT 1920 4:00 - GST +Link Asia/Dubai Asia/Muscat # Oman # Uzbekistan # Byalokoz 1919 says Uzbekistan was 4:27:53. @@ -2874,10 +2868,4 @@ Zone Asia/Ho_Chi_Minh 7:06:40 - LMT 1906 Jul 1 7:00 - ICT # Yemen - -# Milne says 2:59:54 was the meridian of the saluting battery at Aden, -# and that Yemen was at 1:55:56, the meridian of the Hagia Sophia. - -# Zone NAME GMTOFF RULES FORMAT [UNTIL] -Zone Asia/Aden 2:59:54 - LMT 1950 - 3:00 - AST +# See Asia/Riyadh. diff --git a/jdk/test/sun/util/calendar/zi/tzdata/backward b/jdk/test/sun/util/calendar/zi/tzdata/backward index ba012f45733..95266a6f16c 100644 --- a/jdk/test/sun/util/calendar/zi/tzdata/backward +++ b/jdk/test/sun/util/calendar/zi/tzdata/backward @@ -28,7 +28,7 @@ # and their old names. Many names changed in late 1993. # Link TARGET LINK-NAME -Link Africa/Asmara Africa/Asmera +Link Africa/Nairobi Africa/Asmera Link Africa/Abidjan Africa/Timbuktu Link America/Argentina/Catamarca America/Argentina/ComodRivadavia Link America/Adak America/Atka diff --git a/jdk/test/sun/util/calendar/zi/tzdata/europe b/jdk/test/sun/util/calendar/zi/tzdata/europe index 2ed6ad36b5d..89790f06c17 100644 --- a/jdk/test/sun/util/calendar/zi/tzdata/europe +++ b/jdk/test/sun/util/calendar/zi/tzdata/europe @@ -1430,35 +1430,32 @@ Zone Europe/Budapest 1:16:20 - LMT 1890 Oct # might be a reference to the Julian calendar as opposed to Gregorian, or it # might mean something else (???). # -# From Paul Eggert (2006-03-22): -# The Iceland Almanak, Shanks & Pottenger, and Whitman disagree on many points. -# We go with the Almanak, except for one claim from Shanks & Pottenger, namely -# that Reykavik was 21W57 from 1837 to 1908, local mean time before that. +# From Paul Eggert (2014-11-22): +# The information below is taken from the 1988 Almanak; see +# http://www.almanak.hi.is/klukkan.html # # Rule NAME FROM TO TYPE IN ON AT SAVE LETTER/S -Rule Iceland 1917 1918 - Feb 19 23:00 1:00 S +Rule Iceland 1917 1919 - Feb 19 23:00 1:00 S Rule Iceland 1917 only - Oct 21 1:00 0 - -Rule Iceland 1918 only - Nov 16 1:00 0 - +Rule Iceland 1918 1919 - Nov 16 1:00 0 - +Rule Iceland 1921 only - Mar 19 23:00 1:00 S +Rule Iceland 1921 only - Jun 23 1:00 0 - Rule Iceland 1939 only - Apr 29 23:00 1:00 S -Rule Iceland 1939 only - Nov 29 2:00 0 - +Rule Iceland 1939 only - Oct 29 2:00 0 - Rule Iceland 1940 only - Feb 25 2:00 1:00 S -Rule Iceland 1940 only - Nov 3 2:00 0 - -Rule Iceland 1941 only - Mar 2 1:00s 1:00 S -Rule Iceland 1941 only - Nov 2 1:00s 0 - -Rule Iceland 1942 only - Mar 8 1:00s 1:00 S -Rule Iceland 1942 only - Oct 25 1:00s 0 - +Rule Iceland 1940 1941 - Nov Sun>=2 1:00s 0 - +Rule Iceland 1941 1942 - Mar Sun>=2 1:00s 1:00 S # 1943-1946 - first Sunday in March until first Sunday in winter Rule Iceland 1943 1946 - Mar Sun>=1 1:00s 1:00 S -Rule Iceland 1943 1948 - Oct Sun>=22 1:00s 0 - +Rule Iceland 1942 1948 - Oct Sun>=22 1:00s 0 - # 1947-1967 - first Sunday in April until first Sunday in winter Rule Iceland 1947 1967 - Apr Sun>=1 1:00s 1:00 S -# 1949 Oct transition delayed by 1 week +# 1949 and 1967 Oct transitions delayed by 1 week Rule Iceland 1949 only - Oct 30 1:00s 0 - Rule Iceland 1950 1966 - Oct Sun>=22 1:00s 0 - Rule Iceland 1967 only - Oct 29 1:00s 0 - # Zone NAME GMTOFF RULES FORMAT [UNTIL] -Zone Atlantic/Reykjavik -1:27:24 - LMT 1837 - -1:27:48 - RMT 1908 # Reykjavik Mean Time? +Zone Atlantic/Reykjavik -1:28 - LMT 1908 -1:00 Iceland IS%sT 1968 Apr 7 1:00s 0:00 - GMT diff --git a/jdk/test/sun/util/calendar/zi/tzdata/leapseconds b/jdk/test/sun/util/calendar/zi/tzdata/leapseconds index 7612f2bc9b7..9b0a2278433 100644 --- a/jdk/test/sun/util/calendar/zi/tzdata/leapseconds +++ b/jdk/test/sun/util/calendar/zi/tzdata/leapseconds @@ -77,3 +77,7 @@ Leap 1998 Dec 31 23:59:60 + S Leap 2005 Dec 31 23:59:60 + S Leap 2008 Dec 31 23:59:60 + S Leap 2012 Jun 30 23:59:60 + S +Leap 2015 Jun 30 23:59:60 + S + +# Updated through IERS Bulletin C49 +# File expires on: 28 December 2015 diff --git a/jdk/test/sun/util/calendar/zi/tzdata/northamerica b/jdk/test/sun/util/calendar/zi/tzdata/northamerica index 86c9503a4eb..5943cfeec4b 100644 --- a/jdk/test/sun/util/calendar/zi/tzdata/northamerica +++ b/jdk/test/sun/util/calendar/zi/tzdata/northamerica @@ -147,7 +147,7 @@ Rule US 1918 1919 - Mar lastSun 2:00 1:00 D Rule US 1918 1919 - Oct lastSun 2:00 0 S Rule US 1942 only - Feb 9 2:00 1:00 W # War Rule US 1945 only - Aug 14 23:00u 1:00 P # Peace -Rule US 1945 only - Sep 30 2:00 0 S +Rule US 1945 only - Sep lastSun 2:00 0 S Rule US 1967 2006 - Oct lastSun 2:00 0 S Rule US 1967 1973 - Apr lastSun 2:00 1:00 D Rule US 1974 only - Jan 6 2:00 1:00 D @@ -2147,11 +2147,11 @@ Zone America/Dawson -9:17:40 - LMT 1900 Aug 20 # Mexico -# From Paul Eggert (2001-03-05): +# From Paul Eggert (2014-12-07): # The Investigation and Analysis Service of the # Mexican Library of Congress (MLoC) has published a # history of Mexican local time (in Spanish) -# http://www.cddhcu.gob.mx/bibliot/publica/inveyana/polisoc/horver/ +# http://www.diputados.gob.mx/bibliot/publica/inveyana/polisoc/horver/index.htm # # Here are the discrepancies between Shanks & Pottenger (S&P) and the MLoC. # (In all cases we go with the MLoC.) @@ -2320,6 +2320,24 @@ Zone America/Dawson -9:17:40 - LMT 1900 Aug 20 # efecto desde las dos horas del segundo domingo de marzo y concluirá a # las dos horas del primer domingo de noviembre. +# From Steffen Thorsen (2014-12-08), translated by Gwillim Law: +# The Mexican state of Quintana Roo will likely change to EST in 2015. +# +# http://www.unioncancun.mx/articulo/2014/12/04/medio-ambiente/congreso-aprueba-una-hora-mas-de-sol-en-qroo +# "With this change, the time conflict that has existed between the municipios +# of Quintana Roo and the municipio of Felipe Carrillo Puerto may come to an +# end. The latter declared itself in rebellion 15 years ago when a time change +# was initiated in Mexico, and since then it has refused to change its time +# zone along with the rest of the country." +# +# From Steffen Thorsen (2015-01-14), translated by Gwillim Law: +# http://sipse.com/novedades/confirman-aplicacion-de-nueva-zona-horaria-para-quintana-roo-132331.html +# "...the new time zone will come into effect at two o'clock on the first Sunday +# of February, when we will have to advance the clock one hour from its current +# time..." +# +# Also, the new zone will not use DST. + # Rule NAME FROM TO TYPE IN ON AT SAVE LETTER/S Rule Mexico 1939 only - Feb 5 0:00 1:00 D Rule Mexico 1939 only - Jun 25 0:00 0 S @@ -2340,7 +2358,8 @@ Rule Mexico 2002 max - Oct lastSun 2:00 0 S Zone America/Cancun -5:47:04 - LMT 1922 Jan 1 0:12:56 -6:00 - CST 1981 Dec 23 -5:00 Mexico E%sT 1998 Aug 2 2:00 - -6:00 Mexico C%sT + -6:00 Mexico C%sT 2015 Feb 1 2:00 + -5:00 - EST # Campeche, Yucatán; represented by Mérida Zone America/Merida -5:58:28 - LMT 1922 Jan 1 0:01:32 -6:00 - CST 1981 Dec 23 diff --git a/jdk/test/sun/util/calendar/zi/tzdata/southamerica b/jdk/test/sun/util/calendar/zi/tzdata/southamerica index 0b70dea5616..02cf12113a0 100644 --- a/jdk/test/sun/util/calendar/zi/tzdata/southamerica +++ b/jdk/test/sun/util/calendar/zi/tzdata/southamerica @@ -1229,6 +1229,11 @@ Zone America/Rio_Branco -4:31:12 - LMT 1914 # DST Start: first Saturday of September 2014 (Sun 07 Sep 2014 04:00 UTC) # http://www.diariooficial.interior.gob.cl//media/2014/02/19/do-20140219.pdf +# From Juan Correa (2015-01-28): +# ... today the Ministry of Energy announced that Chile will drop DST, will keep +# "summer time" (UTC -3 / UTC -5) all year round.... +# http://www.minenergia.cl/ministerio/noticias/generales/ministerio-de-energia-anuncia.html + # NOTE: ChileAQ rules for Antarctic bases are stored separately in the # 'antarctica' file. @@ -1270,8 +1275,8 @@ Rule Chile 2009 only - Mar Sun>=9 3:00u 0 - Rule Chile 2010 only - Apr Sun>=1 3:00u 0 - Rule Chile 2011 only - May Sun>=2 3:00u 0 - Rule Chile 2011 only - Aug Sun>=16 4:00u 1:00 S -Rule Chile 2012 max - Apr Sun>=23 3:00u 0 - -Rule Chile 2012 max - Sep Sun>=2 4:00u 1:00 S +Rule Chile 2012 2015 - Apr Sun>=23 3:00u 0 - +Rule Chile 2012 2014 - Sep Sun>=2 4:00u 1:00 S # IATA SSIM anomalies: (1992-02) says 1992-03-14; # (1996-09) says 1998-03-08. Ignore these. # Zone NAME GMTOFF RULES FORMAT [UNTIL] @@ -1282,11 +1287,13 @@ Zone America/Santiago -4:42:46 - LMT 1890 -4:00 - CLT 1919 Jul 1 # Chile Time -4:42:46 - SMT 1927 Sep 1 # Santiago Mean Time -5:00 Chile CL%sT 1947 May 22 # Chile Time - -4:00 Chile CL%sT + -4:00 Chile CL%sT 2015 Apr 26 3:00u + -3:00 - CLT Zone Pacific/Easter -7:17:44 - LMT 1890 -7:17:28 - EMT 1932 Sep # Easter Mean Time - -7:00 Chile EAS%sT 1982 Mar 13 21:00 # Easter Time - -6:00 Chile EAS%sT + -7:00 Chile EAS%sT 1982 Mar 13 3:00u # Easter Time + -6:00 Chile EAS%sT 2015 Apr 26 3:00u + -5:00 - EAST # # Salas y Gómez Island is uninhabited. # Other Chilean locations, including Juan Fernández Is, Desventuradas Is, diff --git a/jdk/test/sun/util/calendar/zi/tzdata/zone.tab b/jdk/test/sun/util/calendar/zi/tzdata/zone.tab index 0ef9ba869ea..ffb6469676e 100644 --- a/jdk/test/sun/util/calendar/zi/tzdata/zone.tab +++ b/jdk/test/sun/util/calendar/zi/tzdata/zone.tab @@ -297,7 +297,7 @@ MU -2010+05730 Indian/Mauritius MV +0410+07330 Indian/Maldives MW -1547+03500 Africa/Blantyre MX +1924-09909 America/Mexico_City Central Time - most locations -MX +2105-08646 America/Cancun Central Time - Quintana Roo +MX +2105-08646 America/Cancun Eastern Standard Time - Quintana Roo MX +2058-08937 America/Merida Central Time - Campeche, Yucatan MX +2540-10019 America/Monterrey Mexican Central Time - Coahuila, Durango, Nuevo Leon, Tamaulipas away from US border MX +2550-09730 America/Matamoros US Central Time - Coahuila, Durango, Nuevo Leon, Tamaulipas near US border From 80a1d2bba8a58777d0c423136f67faf3ebe0721c Mon Sep 17 00:00:00 2001 From: Staffan Larsen Date: Thu, 5 Feb 2015 12:59:01 +0100 Subject: [PATCH 47/58] 8072456: @since tags missing from TimeUnit Reviewed-by: alanb, martin --- .../java.base/share/classes/java/util/concurrent/TimeUnit.java | 3 +++ 1 file changed, 3 insertions(+) diff --git a/jdk/src/java.base/share/classes/java/util/concurrent/TimeUnit.java b/jdk/src/java.base/share/classes/java/util/concurrent/TimeUnit.java index e5c007d8c5f..7a8e40b9e43 100644 --- a/jdk/src/java.base/share/classes/java/util/concurrent/TimeUnit.java +++ b/jdk/src/java.base/share/classes/java/util/concurrent/TimeUnit.java @@ -131,6 +131,7 @@ public enum TimeUnit { /** * Time unit representing sixty seconds + * @since 1.6 */ MINUTES { public long toNanos(long d) { return x(d, C4/C0, MAX/(C4/C0)); } @@ -146,6 +147,7 @@ public enum TimeUnit { /** * Time unit representing sixty minutes + * @since 1.6 */ HOURS { public long toNanos(long d) { return x(d, C5/C0, MAX/(C5/C0)); } @@ -161,6 +163,7 @@ public enum TimeUnit { /** * Time unit representing twenty four hours + * @since 1.6 */ DAYS { public long toNanos(long d) { return x(d, C6/C0, MAX/(C6/C0)); } From 4ce8ee0253e7be695c0423a49ef17f61e0fc98d7 Mon Sep 17 00:00:00 2001 From: Staffan Larsen Date: Thu, 5 Feb 2015 13:00:26 +0100 Subject: [PATCH 48/58] 8072458: jdk/test/Makefile references (to be removed) win32 directory in jtreg Reviewed-by: alanb --- jdk/test/Makefile | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/jdk/test/Makefile b/jdk/test/Makefile index 52700cd58d0..792ee4151a6 100644 --- a/jdk/test/Makefile +++ b/jdk/test/Makefile @@ -267,8 +267,8 @@ ifdef CONCURRENCY EXTRA_JTREG_OPTIONS += -concurrency:$(CONCURRENCY) endif -# Default JTREG to run (win32 script works for everybody) -JTREG = $(JT_HOME)/win32/bin/jtreg +# Default JTREG to run +JTREG = $(JT_HOME)/bin/jtreg # run in agentvm mode JTREG_BASIC_OPTIONS += -agentvm # Only run automatic tests From 2d266be7f6de843ef1801b8de5863af5b71be19b Mon Sep 17 00:00:00 2001 From: Athijegannathan Sundararajan Date: Thu, 5 Feb 2015 19:08:00 +0530 Subject: [PATCH 49/58] 8072595: nashorn should not use obj.getClass() for null checks Reviewed-by: hannesw, attila --- nashorn/samples/javashell.js | 3 ++- nashorn/samples/shell.js | 3 ++- .../jdk/internal/dynalink/DynamicLinker.java | 3 ++- .../internal/dynalink/beans/StaticClass.java | 4 ++-- .../dynalink/linker/GuardedInvocation.java | 7 +++---- .../support/CallSiteDescriptorFactory.java | 7 ++++--- .../api/scripting/NashornScriptEngine.java | 3 ++- .../scripting/NashornScriptEngineFactory.java | 11 +++++----- .../api/scripting/ScriptObjectMirror.java | 21 +++++++++---------- .../jdk/nashorn/api/scripting/URLReader.java | 4 ++-- .../nashorn/internal/codegen/CompileUnit.java | 3 ++- .../jdk/nashorn/internal/objects/Global.java | 6 +++--- .../jdk/nashorn/internal/runtime/Context.java | 3 ++- .../internal/runtime/ScriptLoader.java | 4 ++-- .../runtime/linker/JavaSuperAdapter.java | 4 +++- .../internal/runtime/options/Options.java | 7 ++++--- .../api/scripting/ScriptEngineTest.java | 9 ++++++++ 17 files changed, 60 insertions(+), 42 deletions(-) diff --git a/nashorn/samples/javashell.js b/nashorn/samples/javashell.js index 4e71f753914..3fac42b17ec 100644 --- a/nashorn/samples/javashell.js +++ b/nashorn/samples/javashell.js @@ -40,6 +40,7 @@ var Arrays = Java.type("java.util.Arrays"); var BufferedReader = Java.type("java.io.BufferedReader"); var FileWriter = Java.type("java.io.FileWriter"); +var List = Java.type("java.util.List"); var LocalDateTime = Java.type("java.time.LocalDateTime"); var InputStreamReader = Java.type("java.io.InputStreamReader"); var PrintWriter = Java.type("java.io.PrintWriter"); @@ -121,7 +122,7 @@ EOF // execute code command function exec(args) { // build child process and start it! - new ProcessBuilder(Arrays.asList(args.split(' '))) + new ProcessBuilder(Java.to(args.split(' '), List)) .inheritIO() .start() .waitFor(); diff --git a/nashorn/samples/shell.js b/nashorn/samples/shell.js index 8e3195ce049..65e508b3217 100644 --- a/nashorn/samples/shell.js +++ b/nashorn/samples/shell.js @@ -42,6 +42,7 @@ var Arrays = Java.type("java.util.Arrays"); var BufferedReader = Java.type("java.io.BufferedReader"); var InputStreamReader = Java.type("java.io.InputStreamReader"); + var List = Java.type("java.util.List"); var ProcessBuilder = Java.type("java.lang.ProcessBuilder"); var System = Java.type("java.lang.System"); @@ -66,7 +67,7 @@ } } else { // build child process and start it! - new ProcessBuilder(Arrays.asList(args)) + new ProcessBuilder(Java.to(args, List)) .inheritIO() .start() .waitFor(); diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/internal/dynalink/DynamicLinker.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/internal/dynalink/DynamicLinker.java index e7e58d2a8f4..289710e2f2a 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/internal/dynalink/DynamicLinker.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/internal/dynalink/DynamicLinker.java @@ -88,6 +88,7 @@ import java.lang.invoke.MethodHandles; import java.lang.invoke.MethodType; import java.lang.invoke.MutableCallSite; import java.util.List; +import java.util.Objects; import jdk.internal.dynalink.linker.GuardedInvocation; import jdk.internal.dynalink.linker.GuardingDynamicLinker; import jdk.internal.dynalink.linker.LinkRequest; @@ -252,7 +253,7 @@ public class DynamicLinker { // Make sure we filter the invocation before linking it into the call site. This is typically used to match the // return type of the invocation to the call site. guardedInvocation = prelinkFilter.filter(guardedInvocation, linkRequest, linkerServices); - guardedInvocation.getClass(); // null pointer check + Objects.requireNonNull(guardedInvocation); int newRelinkCount = relinkCount; // Note that the short-circuited "&&" evaluation below ensures we'll increment the relinkCount until diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/internal/dynalink/beans/StaticClass.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/internal/dynalink/beans/StaticClass.java index f70f264709b..153c12cfbaa 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/internal/dynalink/beans/StaticClass.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/internal/dynalink/beans/StaticClass.java @@ -84,6 +84,7 @@ package jdk.internal.dynalink.beans; import java.io.Serializable; +import java.util.Objects; /** * Object that represents the static facet of a class (its static methods, properties, and fields, as well as @@ -106,8 +107,7 @@ public class StaticClass implements Serializable { private final Class clazz; /*private*/ StaticClass(final Class clazz) { - clazz.getClass(); // NPE check - this.clazz = clazz; + this.clazz = Objects.requireNonNull(clazz); } /** diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/internal/dynalink/linker/GuardedInvocation.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/internal/dynalink/linker/GuardedInvocation.java index 00ee9dc77cb..c5357ac3cf9 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/internal/dynalink/linker/GuardedInvocation.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/internal/dynalink/linker/GuardedInvocation.java @@ -91,6 +91,7 @@ import java.lang.invoke.MethodType; import java.lang.invoke.SwitchPoint; import java.lang.invoke.WrongMethodTypeException; import java.util.List; +import java.util.Objects; import jdk.internal.dynalink.CallSiteDescriptor; import jdk.internal.dynalink.support.Guards; @@ -170,8 +171,7 @@ public class GuardedInvocation { * @throws NullPointerException if invocation is null. */ public GuardedInvocation(final MethodHandle invocation, final MethodHandle guard, final SwitchPoint switchPoint, final Class exception) { - invocation.getClass(); // NPE check - this.invocation = invocation; + this.invocation = Objects.requireNonNull(invocation); this.guard = guard; this.switchPoints = switchPoint == null ? null : new SwitchPoint[] { switchPoint }; this.exception = exception; @@ -190,8 +190,7 @@ public class GuardedInvocation { * @throws NullPointerException if invocation is null. */ public GuardedInvocation(final MethodHandle invocation, final MethodHandle guard, final SwitchPoint[] switchPoints, final Class exception) { - invocation.getClass(); // NPE check - this.invocation = invocation; + this.invocation = Objects.requireNonNull(invocation); this.guard = guard; this.switchPoints = switchPoints == null ? null : switchPoints.clone(); this.exception = exception; diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/internal/dynalink/support/CallSiteDescriptorFactory.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/internal/dynalink/support/CallSiteDescriptorFactory.java index 9cbd5f04f3f..bfc0dea1543 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/internal/dynalink/support/CallSiteDescriptorFactory.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/internal/dynalink/support/CallSiteDescriptorFactory.java @@ -91,6 +91,7 @@ import java.lang.ref.WeakReference; import java.util.Arrays; import java.util.Collections; import java.util.List; +import java.util.Objects; import java.util.StringTokenizer; import java.util.WeakHashMap; import jdk.internal.dynalink.CallSiteDescriptor; @@ -123,9 +124,9 @@ public class CallSiteDescriptorFactory { * in fact return a weakly-referenced canonical instance. */ public static CallSiteDescriptor create(final Lookup lookup, final String name, final MethodType methodType) { - name.getClass(); // NPE check - methodType.getClass(); // NPE check - lookup.getClass(); // NPE check + Objects.requireNonNull(name); + Objects.requireNonNull(methodType); + Objects.requireNonNull(lookup); final String[] tokenizedName = tokenizeName(name); if(isPublicLookup(lookup)) { return getCanonicalPublicDescriptor(createPublicCallSiteDescriptor(tokenizedName, methodType)); diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/api/scripting/NashornScriptEngine.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/api/scripting/NashornScriptEngine.java index 46925c4f3f6..dc5170690c3 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/api/scripting/NashornScriptEngine.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/api/scripting/NashornScriptEngine.java @@ -39,6 +39,7 @@ import java.security.PrivilegedAction; import java.security.ProtectionDomain; import java.text.MessageFormat; import java.util.Locale; +import java.util.Objects; import java.util.ResourceBundle; import javax.script.AbstractScriptEngine; import javax.script.Bindings; @@ -360,7 +361,7 @@ public final class NashornScriptEngine extends AbstractScriptEngine implements C } private Object invokeImpl(final Object selfObject, final String name, final Object... args) throws ScriptException, NoSuchMethodException { - name.getClass(); // null check + Objects.requireNonNull(name); assert !(selfObject instanceof ScriptObject) : "raw ScriptObject not expected here"; Global invokeGlobal = null; diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/api/scripting/NashornScriptEngineFactory.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/api/scripting/NashornScriptEngineFactory.java index 7201b9410b2..412d4458765 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/api/scripting/NashornScriptEngineFactory.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/api/scripting/NashornScriptEngineFactory.java @@ -28,6 +28,7 @@ package jdk.nashorn.api.scripting; import java.util.Arrays; import java.util.Collections; import java.util.List; +import java.util.Objects; import javax.script.ScriptEngine; import javax.script.ScriptEngineFactory; import jdk.nashorn.internal.runtime.Context; @@ -177,7 +178,7 @@ public final class NashornScriptEngineFactory implements ScriptEngineFactory { * denies {@code RuntimePermission("nashorn.setConfig")} */ public ScriptEngine getScriptEngine(final ClassFilter classFilter) { - classFilter.getClass(); // null check + Objects.requireNonNull(classFilter); return newEngine(DEFAULT_OPTIONS, getAppClassLoader(), classFilter); } @@ -192,7 +193,7 @@ public final class NashornScriptEngineFactory implements ScriptEngineFactory { * denies {@code RuntimePermission("nashorn.setConfig")} */ public ScriptEngine getScriptEngine(final String... args) { - args.getClass(); // null check + Objects.requireNonNull(args); return newEngine(args, getAppClassLoader(), null); } @@ -208,7 +209,7 @@ public final class NashornScriptEngineFactory implements ScriptEngineFactory { * denies {@code RuntimePermission("nashorn.setConfig")} */ public ScriptEngine getScriptEngine(final String[] args, final ClassLoader appLoader) { - args.getClass(); // null check + Objects.requireNonNull(args); return newEngine(args, appLoader, null); } @@ -225,8 +226,8 @@ public final class NashornScriptEngineFactory implements ScriptEngineFactory { * denies {@code RuntimePermission("nashorn.setConfig")} */ public ScriptEngine getScriptEngine(final String[] args, final ClassLoader appLoader, final ClassFilter classFilter) { - args.getClass(); // null check - classFilter.getClass(); // null check + Objects.requireNonNull(args); + Objects.requireNonNull(classFilter); return newEngine(args, appLoader, classFilter); } diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/api/scripting/ScriptObjectMirror.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/api/scripting/ScriptObjectMirror.java index d18316e81f6..f200fee0709 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/api/scripting/ScriptObjectMirror.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/api/scripting/ScriptObjectMirror.java @@ -39,6 +39,7 @@ import java.util.Iterator; import java.util.LinkedHashSet; import java.util.List; import java.util.Map; +import java.util.Objects; import java.util.Set; import java.util.concurrent.Callable; import javax.script.Bindings; @@ -180,7 +181,7 @@ public final class ScriptObjectMirror extends AbstractJSObject implements Bindin * @return return value of function */ public Object callMember(final String functionName, final Object... args) { - functionName.getClass(); // null check + Objects.requireNonNull(functionName); final Global oldGlobal = Context.getGlobal(); final boolean globalChanged = (oldGlobal != global); @@ -213,7 +214,7 @@ public final class ScriptObjectMirror extends AbstractJSObject implements Bindin @Override public Object getMember(final String name) { - name.getClass(); + Objects.requireNonNull(name); return inGlobal(new Callable() { @Override public Object call() { return wrap(sobj.get(name), global); @@ -232,7 +233,7 @@ public final class ScriptObjectMirror extends AbstractJSObject implements Bindin @Override public boolean hasMember(final String name) { - name.getClass(); + Objects.requireNonNull(name); return inGlobal(new Callable() { @Override public Boolean call() { return sobj.has(name); @@ -251,13 +252,13 @@ public final class ScriptObjectMirror extends AbstractJSObject implements Bindin @Override public void removeMember(final String name) { - name.getClass(); + Objects.requireNonNull(name); remove(name); } @Override public void setMember(final String name, final Object value) { - name.getClass(); + Objects.requireNonNull(name); put(name, value); } @@ -425,9 +426,7 @@ public final class ScriptObjectMirror extends AbstractJSObject implements Bindin @Override public void putAll(final Map map) { - if (map == null) { - throw new NullPointerException("map is null"); - } + Objects.requireNonNull(map, "map is null"); final ScriptObject oldGlobal = Context.getGlobal(); final boolean globalChanged = (oldGlobal != global); inGlobal(new Callable() { @@ -804,9 +803,9 @@ public final class ScriptObjectMirror extends AbstractJSObject implements Bindin * @throws IllegalArgumentException if key is empty string */ private static void checkKey(final Object key) { - if (key == null) { - throw new NullPointerException("key can not be null"); - } else if (!(key instanceof String)) { + Objects.requireNonNull(key, "key can not be null"); + + if (!(key instanceof String)) { throw new ClassCastException("key should be a String. It is " + key.getClass().getName() + " instead."); } else if (((String)key).length() == 0) { throw new IllegalArgumentException("key can not be empty"); diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/api/scripting/URLReader.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/api/scripting/URLReader.java index bd16629d6c8..6ba848833dc 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/api/scripting/URLReader.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/api/scripting/URLReader.java @@ -30,6 +30,7 @@ import java.io.IOException; import java.io.Reader; import java.net.URL; import java.nio.charset.Charset; +import java.util.Objects; import jdk.nashorn.internal.runtime.Source; /** @@ -77,8 +78,7 @@ public final class URLReader extends Reader { * @throws NullPointerException if url is null */ public URLReader(final URL url, final Charset cs) { - // null check - url.getClass(); + Objects.requireNonNull(url); this.url = url; this.cs = cs; } diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/codegen/CompileUnit.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/codegen/CompileUnit.java index 13e9d7168cb..a428e5bc5d2 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/codegen/CompileUnit.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/codegen/CompileUnit.java @@ -26,6 +26,7 @@ package jdk.nashorn.internal.codegen; import java.io.Serializable; +import java.util.Objects; import java.util.Set; import java.util.TreeSet; import jdk.nashorn.internal.ir.CompileUnitHolder; @@ -113,7 +114,7 @@ public final class CompileUnit implements Comparable, Serializable * @param clazz class with code for this compile unit */ void setCode(final Class clazz) { - clazz.getClass(); // null check + Objects.requireNonNull(clazz); this.clazz = clazz; // Revisit this - refactor to avoid null-ed out non-final fields // null out emitter diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/objects/Global.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/objects/Global.java index 6418eaf561c..8c3f331d250 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/objects/Global.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/objects/Global.java @@ -41,6 +41,7 @@ import java.util.ArrayList; import java.util.Arrays; import java.util.List; import java.util.Map; +import java.util.Objects; import java.util.concurrent.Callable; import java.util.concurrent.ConcurrentHashMap; import javax.script.ScriptContext; @@ -463,8 +464,7 @@ public final class Global extends ScriptObject implements Scope { sm.checkPermission(new RuntimePermission(Context.NASHORN_CREATE_GLOBAL)); } - // null check on context - context.getClass(); + Objects.requireNonNull(context); return $nasgenmap$; } @@ -488,7 +488,7 @@ public final class Global extends ScriptObject implements Scope { */ public static Global instance() { final Global global = Context.getGlobal(); - global.getClass(); // null check + Objects.requireNonNull(global); return global; } diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/Context.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/Context.java index 5a791dc88c7..8c265c8a7ff 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/Context.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/Context.java @@ -60,6 +60,7 @@ import java.util.Collection; import java.util.HashMap; import java.util.LinkedHashMap; import java.util.Map; +import java.util.Objects; import java.util.concurrent.atomic.AtomicLong; import java.util.concurrent.atomic.AtomicReference; import java.util.function.Consumer; @@ -904,7 +905,7 @@ public final class Context { * @throw SecurityException if not accessible */ private static void checkPackageAccess(final SecurityManager sm, final String fullName) { - sm.getClass(); // null check + Objects.requireNonNull(sm); final int index = fullName.lastIndexOf('.'); if (index != -1) { final String pkgName = fullName.substring(0, index); diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/ScriptLoader.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/ScriptLoader.java index b2a54beb3f5..4dabbbacaa4 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/ScriptLoader.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/ScriptLoader.java @@ -26,6 +26,7 @@ package jdk.nashorn.internal.runtime; import java.security.CodeSource; +import java.util.Objects; /** * Responsible for loading script generated classes. @@ -69,8 +70,7 @@ final class ScriptLoader extends NashornLoader { * @return Installed class. */ synchronized Class installClass(final String name, final byte[] data, final CodeSource cs) { - // null check - cs.getClass(); + Objects.requireNonNull(cs); return defineClass(name, data, 0, data.length, cs); } } diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/linker/JavaSuperAdapter.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/linker/JavaSuperAdapter.java index 3c7fcb26a42..94f2a5109d2 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/linker/JavaSuperAdapter.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/linker/JavaSuperAdapter.java @@ -25,6 +25,8 @@ package jdk.nashorn.internal.runtime.linker; +import java.util.Objects; + /** * Represents a an adapter for invoking superclass methods on an adapter instance generated by * {@code JavaAdapterBytecodeGenerator}. Note that objects of this class are just wrappers around the adapter instances, @@ -34,7 +36,7 @@ class JavaSuperAdapter { private final Object adapter; JavaSuperAdapter(final Object adapter) { - adapter.getClass(); // NPE check + Objects.requireNonNull(adapter); this.adapter = adapter; } diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/options/Options.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/options/Options.java index ed39fa95481..6cb68abe3c3 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/options/Options.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/options/Options.java @@ -42,6 +42,7 @@ import java.util.List; import java.util.Locale; import java.util.Map; import java.util.MissingResourceException; +import java.util.Objects; import java.util.PropertyPermission; import java.util.ResourceBundle; import java.util.StringTokenizer; @@ -143,7 +144,7 @@ public final class Options { * @return true if set to true, default value if unset or set to false */ public static boolean getBooleanProperty(final String name, final Boolean defValue) { - name.getClass(); // null check + Objects.requireNonNull(name); if (!name.startsWith("nashorn.")) { throw new IllegalArgumentException(name); } @@ -184,7 +185,7 @@ public final class Options { * @return string property if set or default value */ public static String getStringProperty(final String name, final String defValue) { - name.getClass(); // null check + Objects.requireNonNull(name); if (! name.startsWith("nashorn.")) { throw new IllegalArgumentException(name); } @@ -211,7 +212,7 @@ public final class Options { * @return integer property if set or default value */ public static int getIntProperty(final String name, final int defValue) { - name.getClass(); // null check + Objects.requireNonNull(name); if (! name.startsWith("nashorn.")) { throw new IllegalArgumentException(name); } diff --git a/nashorn/test/src/jdk/nashorn/api/scripting/ScriptEngineTest.java b/nashorn/test/src/jdk/nashorn/api/scripting/ScriptEngineTest.java index a32e2e95df8..7a0a346207a 100644 --- a/nashorn/test/src/jdk/nashorn/api/scripting/ScriptEngineTest.java +++ b/nashorn/test/src/jdk/nashorn/api/scripting/ScriptEngineTest.java @@ -721,6 +721,15 @@ public class ScriptEngineTest { assertTrue(invoked.get()); } + @Test + public void testLengthOnArrayLikeObjects() throws Exception { + final ScriptEngine e = new ScriptEngineManager().getEngineByName("nashorn"); + final Object val = e.eval("var arr = { length: 1, 0: 1}; arr.length"); + + assertTrue(Number.class.isAssignableFrom(val.getClass())); + assertTrue(((Number)val).intValue() == 1); + } + // @bug JDK-8068603: NashornScriptEngine.put/get() impls don't conform to NPE, IAE spec assertions @Test public void illegalBindingsValuesTest() throws Exception { From d4e6353375bbec7a87b10d7069fd8b6dfb7eb746 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Hannes=20Walln=C3=B6fer?= Date: Thu, 5 Feb 2015 14:42:14 +0100 Subject: [PATCH 50/58] 8062141: Various performance issues parsing JSON Reviewed-by: lagergren, attila --- .../jdk/nashorn/internal/objects/Global.java | 23 +- .../nashorn/internal/parser/JSONParser.java | 716 ++++++++++-------- .../jdk/nashorn/internal/parser/Lexer.java | 21 - .../internal/runtime/JSONFunctions.java | 87 +-- .../nashorn/internal/runtime/PropertyMap.java | 2 +- .../jdk/nashorn/internal/scripts/JO.java | 6 +- nashorn/test/examples/json-parser-micro.js | 315 ++++++++ nashorn/test/script/basic/JDK-8062141.js | 91 +++ .../test/script/basic/JDK-8062141.js.EXPECTED | 120 +++ .../test/script/basic/NASHORN-623.js.EXPECTED | 4 +- 10 files changed, 961 insertions(+), 424 deletions(-) create mode 100644 nashorn/test/examples/json-parser-micro.js create mode 100644 nashorn/test/script/basic/JDK-8062141.js create mode 100644 nashorn/test/script/basic/JDK-8062141.js.EXPECTED diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/objects/Global.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/objects/Global.java index 8c3f331d250..4ff5123db55 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/objects/Global.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/objects/Global.java @@ -580,13 +580,15 @@ public final class Global extends ScriptObject implements Scope { } else if (obj instanceof String || obj instanceof ConsString) { return new NativeString((CharSequence)obj, this); } else if (obj instanceof Object[]) { // extension - return new NativeArray((Object[])obj); + return new NativeArray(ArrayData.allocate((Object[])obj), this); } else if (obj instanceof double[]) { // extension - return new NativeArray((double[])obj); + return new NativeArray(ArrayData.allocate((double[])obj), this); } else if (obj instanceof long[]) { - return new NativeArray((long[])obj); + return new NativeArray(ArrayData.allocate((long[])obj), this); } else if (obj instanceof int[]) { - return new NativeArray((int[])obj); + return new NativeArray(ArrayData.allocate((int[]) obj), this); + } else if (obj instanceof ArrayData) { + return new NativeArray((ArrayData) obj, this); } else { // FIXME: more special cases? Map? List? return obj; @@ -1028,12 +1030,17 @@ public final class Global extends ScriptObject implements Scope { } // builtin prototype accessors - ScriptObject getFunctionPrototype() { - return ScriptFunction.getPrototype(builtinFunction); + + /** + * Get the builtin Object prototype. + * @return the object prototype. + */ + public ScriptObject getObjectPrototype() { + return ScriptFunction.getPrototype(builtinObject); } - ScriptObject getObjectPrototype() { - return ScriptFunction.getPrototype(builtinObject); + ScriptObject getFunctionPrototype() { + return ScriptFunction.getPrototype(builtinFunction); } ScriptObject getArrayPrototype() { diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/parser/JSONParser.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/parser/JSONParser.java index 7faf89a80c7..e11a9efac71 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/parser/JSONParser.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/parser/JSONParser.java @@ -1,5 +1,5 @@ /* - * Copyright (c) 2010, 2013, Oracle and/or its affiliates. All rights reserved. + * Copyright (c) 2010, 2015, 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 @@ -25,38 +25,59 @@ package jdk.nashorn.internal.parser; -import static jdk.nashorn.internal.parser.TokenType.COLON; -import static jdk.nashorn.internal.parser.TokenType.COMMARIGHT; -import static jdk.nashorn.internal.parser.TokenType.EOF; -import static jdk.nashorn.internal.parser.TokenType.ESCSTRING; -import static jdk.nashorn.internal.parser.TokenType.RBRACE; -import static jdk.nashorn.internal.parser.TokenType.RBRACKET; -import static jdk.nashorn.internal.parser.TokenType.STRING; import java.util.ArrayList; import java.util.List; -import jdk.nashorn.internal.ir.Expression; -import jdk.nashorn.internal.ir.LiteralNode; -import jdk.nashorn.internal.ir.Node; -import jdk.nashorn.internal.ir.ObjectNode; -import jdk.nashorn.internal.ir.PropertyNode; -import jdk.nashorn.internal.ir.UnaryNode; +import jdk.nashorn.internal.codegen.ObjectClassGenerator; +import jdk.nashorn.internal.objects.Global; +import jdk.nashorn.internal.runtime.ECMAErrors; import jdk.nashorn.internal.runtime.ErrorManager; +import jdk.nashorn.internal.runtime.JSErrorType; +import jdk.nashorn.internal.runtime.JSType; +import jdk.nashorn.internal.runtime.ParserException; +import jdk.nashorn.internal.runtime.Property; +import jdk.nashorn.internal.runtime.PropertyMap; +import jdk.nashorn.internal.runtime.ScriptObject; import jdk.nashorn.internal.runtime.Source; +import jdk.nashorn.internal.runtime.SpillProperty; +import jdk.nashorn.internal.runtime.arrays.ArrayData; +import jdk.nashorn.internal.runtime.arrays.ArrayIndex; +import jdk.nashorn.internal.scripts.JO; + +import static jdk.nashorn.internal.parser.TokenType.STRING; /** * Parses JSON text and returns the corresponding IR node. This is derived from the objectLiteral production of the main parser. * * See: 15.12.1.2 The JSON Syntactic Grammar */ -public class JSONParser extends AbstractParser { +public class JSONParser { + + final private String source; + final private Global global; + final int length; + int pos = 0; + + private static PropertyMap EMPTY_MAP = PropertyMap.newMap(); + + private static final int EOF = -1; + + private static final String TRUE = "true"; + private static final String FALSE = "false"; + private static final String NULL = "null"; + + private static final int STATE_EMPTY = 0; + private static final int STATE_ELEMENT_PARSED = 1; + private static final int STATE_COMMA_PARSED = 2; /** * Constructor * @param source the source - * @param errors the error manager + * @param global the global object */ - public JSONParser(final Source source, final ErrorManager errors) { - super(source, errors, false, 0); + public JSONParser(final String source, final Global global ) { + this.source = source; + this.global = global; + this.length = source.length(); } /** @@ -114,329 +135,408 @@ public class JSONParser extends AbstractParser { } /** - * Public parsed method - start lexing a new token stream for - * a JSON script + * Public parse method. Parse a string into a JSON object. * - * @return the JSON literal + * @return the parsed JSON Object */ - public Node parse() { - stream = new TokenStream(); - - lexer = new Lexer(source, stream) { - - @Override - protected boolean skipComments() { - return false; - } - - @Override - protected boolean isStringDelimiter(final char ch) { - return ch == '\"'; - } - - // ECMA 15.12.1.1 The JSON Lexical Grammar - JSONWhiteSpace - @Override - protected boolean isWhitespace(final char ch) { - return Lexer.isJsonWhitespace(ch); - } - - @Override - protected boolean isEOL(final char ch) { - return Lexer.isJsonEOL(ch); - } - - // ECMA 15.12.1.1 The JSON Lexical Grammar - JSONNumber - @Override - protected void scanNumber() { - // Record beginning of number. - final int startPosition = position; - // Assume value is a decimal. - TokenType valueType = TokenType.DECIMAL; - - // floating point can't start with a "." with no leading digit before - if (ch0 == '.') { - error(Lexer.message("json.invalid.number"), STRING, position, limit); - } - - // First digit of number. - final int digit = convertDigit(ch0, 10); - - // skip first digit - skip(1); - - if (digit != 0) { - // Skip over remaining digits. - while (convertDigit(ch0, 10) != -1) { - skip(1); - } - } - - if (ch0 == '.' || ch0 == 'E' || ch0 == 'e') { - // Must be a double. - if (ch0 == '.') { - // Skip period. - skip(1); - - boolean mantissa = false; - // Skip mantissa. - while (convertDigit(ch0, 10) != -1) { - mantissa = true; - skip(1); - } - - if (! mantissa) { - // no digit after "." - error(Lexer.message("json.invalid.number"), STRING, position, limit); - } - } - - // Detect exponent. - if (ch0 == 'E' || ch0 == 'e') { - // Skip E. - skip(1); - // Detect and skip exponent sign. - if (ch0 == '+' || ch0 == '-') { - skip(1); - } - boolean exponent = false; - // Skip exponent. - while (convertDigit(ch0, 10) != -1) { - exponent = true; - skip(1); - } - - if (! exponent) { - // no digit after "E" - error(Lexer.message("json.invalid.number"), STRING, position, limit); - } - } - - valueType = TokenType.FLOATING; - } - - // Add number token. - add(valueType, startPosition); - } - - // ECMA 15.12.1.1 The JSON Lexical Grammar - JSONEscapeCharacter - @Override - protected boolean isEscapeCharacter(final char ch) { - switch (ch) { - case '"': - case '/': - case '\\': - case 'b': - case 'f': - case 'n': - case 'r': - case 't': - // could be unicode escape - case 'u': - return true; - default: - return false; - } - } - }; - - k = -1; - - next(); - - final Node resultNode = jsonLiteral(); - expect(EOF); - - return resultNode; - } - - @SuppressWarnings("fallthrough") - private LiteralNode getStringLiteral() { - final LiteralNode literal = getLiteral(); - final String str = (String)literal.getValue(); - - for (int i = 0; i < str.length(); i++) { - final char ch = str.charAt(i); - switch (ch) { - default: - if (ch > 0x001f) { - break; - } - case '"': - case '\\': - throw error(AbstractParser.message("unexpected.token", str)); - } + public Object parse() { + final Object value = parseLiteral(); + skipWhiteSpace(); + if (pos < length) { + throw expectedError(pos, "eof", toString(peek())); } - - return literal; + return value; } - /** - * Parse a JSON literal from the token stream - * @return the JSON literal as a Node - */ - private Expression jsonLiteral() { - final long literalToken = token; + private Object parseLiteral() { + skipWhiteSpace(); - switch (type) { - case STRING: - return getStringLiteral(); - case ESCSTRING: - case DECIMAL: - case FLOATING: - return getLiteral(); - case FALSE: - next(); - return LiteralNode.newInstance(literalToken, finish, false); - case TRUE: - next(); - return LiteralNode.newInstance(literalToken, finish, true); - case NULL: - next(); - return LiteralNode.newInstance(literalToken, finish); - case LBRACKET: - return arrayLiteral(); - case LBRACE: - return objectLiteral(); - /* - * A.8.1 JSON Lexical Grammar - * - * JSONNumber :: See 15.12.1.1 - * -opt DecimalIntegerLiteral JSONFractionopt ExponentPartopt - */ - case SUB: - next(); - - final long realToken = token; - final Object value = getValue(); - - if (value instanceof Number) { - next(); - return new UnaryNode(literalToken, LiteralNode.newInstance(realToken, finish, (Number)value)); - } - - throw error(AbstractParser.message("expected", "number", type.getNameOrType())); + final int c = peek(); + if (c == EOF) { + throw expectedError(pos, "json literal", "eof"); + } + switch (c) { + case '{': + return parseObject(); + case '[': + return parseArray(); + case '"': + return parseString(); + case 'f': + return parseKeyword(FALSE, Boolean.FALSE); + case 't': + return parseKeyword(TRUE, Boolean.TRUE); + case 'n': + return parseKeyword(NULL, null); default: - break; + if (isDigit(c) || c == '-') { + return parseNumber(); + } else if (c == '.') { + throw numberError(pos); + } else { + throw expectedError(pos, "json literal", toString(c)); + } } - - throw error(AbstractParser.message("expected", "json literal", type.getNameOrType())); } - /** - * Parse an array literal from the token stream - * @return the array literal as a Node - */ - private LiteralNode arrayLiteral() { - // Unlike JavaScript array literals, elison is not permitted in JSON. + private Object parseObject() { + PropertyMap propertyMap = EMPTY_MAP; + ArrayData arrayData = ArrayData.EMPTY_ARRAY; + final ArrayList values = new ArrayList<>(); + int state = STATE_EMPTY; - // Capture LBRACKET token. - final long arrayToken = token; - // LBRACKET tested in caller. - next(); + assert peek() == '{'; + pos++; - LiteralNode result = null; - // Prepare to accummulating elements. - final List elements = new ArrayList<>(); + while (pos < length) { + skipWhiteSpace(); + final int c = peek(); -loop: - while (true) { - switch (type) { - case RBRACKET: - next(); - result = LiteralNode.newInstance(arrayToken, finish, elements); - break loop; - - case COMMARIGHT: - next(); - // check for trailing comma - not allowed in JSON - if (type == RBRACKET) { - throw error(AbstractParser.message("trailing.comma.in.json", type.getNameOrType())); + switch (c) { + case '"': + if (state == STATE_ELEMENT_PARSED) { + throw expectedError(pos - 1, ", or }", toString(c)); } + final String id = parseString(); + expectColon(); + final Object value = parseLiteral(); + final int index = ArrayIndex.getArrayIndex(id); + if (ArrayIndex.isValidArrayIndex(index)) { + arrayData = addArrayElement(arrayData, index, value); + } else { + propertyMap = addObjectProperty(propertyMap, values, id, value); + } + state = STATE_ELEMENT_PARSED; break; - - default: - // Add expression element. - elements.add(jsonLiteral()); - // Comma between array elements is mandatory in JSON. - if (type != COMMARIGHT && type != RBRACKET) { - throw error(AbstractParser.message("expected", ", or ]", type.getNameOrType())); + case ',': + if (state != STATE_ELEMENT_PARSED) { + throw error(AbstractParser.message("trailing.comma.in.json"), pos); } + state = STATE_COMMA_PARSED; + pos++; + break; + case '}': + if (state == STATE_COMMA_PARSED) { + throw error(AbstractParser.message("trailing.comma.in.json"), pos); + } + pos++; + return createObject(propertyMap, values, arrayData); + default: + throw expectedError(pos, ", or }", toString(c)); + } + } + throw expectedError(pos, ", or }", "eof"); + } + + private static ArrayData addArrayElement(final ArrayData arrayData, final int index, final Object value) { + final long oldLength = arrayData.length(); + final long longIndex = ArrayIndex.toLongIndex(index); + ArrayData newArrayData = arrayData; + if (longIndex > oldLength) { + if (arrayData.canDelete(oldLength, longIndex - 1, false)) { + newArrayData = newArrayData.delete(oldLength, longIndex - 1); + } + } + return newArrayData.ensure(longIndex).set(index, value, false); + } + + private static PropertyMap addObjectProperty(final PropertyMap propertyMap, final List values, + final String id, final Object value) { + final Property oldProperty = propertyMap.findProperty(id); + final Property newProperty; + final PropertyMap newMap; + final Class type = ObjectClassGenerator.OBJECT_FIELDS_ONLY ? Object.class : getType(value); + + if (oldProperty != null) { + values.set(oldProperty.getSlot(), value); + newProperty = new SpillProperty(id, 0, oldProperty.getSlot()); + newProperty.setType(type); + newMap = propertyMap.replaceProperty(oldProperty, newProperty);; + } else { + values.add(value); + newProperty = new SpillProperty(id, 0, propertyMap.size()); + newProperty.setType(type); + newMap = propertyMap.addProperty(newProperty); + } + + return newMap; + } + + private Object createObject(final PropertyMap propertyMap, final List values, final ArrayData arrayData) { + final long[] primitiveSpill = new long[values.size()]; + final Object[] objectSpill = new Object[values.size()]; + + for (final Property property : propertyMap.getProperties()) { + if (property.getType() == Object.class) { + objectSpill[property.getSlot()] = values.get(property.getSlot()); + } else { + primitiveSpill[property.getSlot()] = ObjectClassGenerator.pack((Number) values.get(property.getSlot())); + } + } + + final ScriptObject object = new JO(propertyMap, primitiveSpill, objectSpill); + object.setInitialProto(global.getObjectPrototype()); + object.setArray(arrayData); + return object; + } + + private static Class getType(final Object value) { + if (value instanceof Integer) { + return int.class; + } else if (value instanceof Long) { + return long.class; + } else if (value instanceof Double) { + return double.class; + } else { + return Object.class; + } + } + + private void expectColon() { + skipWhiteSpace(); + final int n = next(); + if (n != ':') { + throw expectedError(pos - 1, ":", toString(n)); + } + } + + private Object parseArray() { + ArrayData arrayData = ArrayData.EMPTY_ARRAY; + int state = STATE_EMPTY; + + assert peek() == '['; + pos++; + + while (pos < length) { + skipWhiteSpace(); + final int c = peek(); + + switch (c) { + case ',': + if (state != STATE_ELEMENT_PARSED) { + throw error(AbstractParser.message("trailing.comma.in.json"), pos); + } + state = STATE_COMMA_PARSED; + pos++; + break; + case ']': + if (state == STATE_COMMA_PARSED) { + throw error(AbstractParser.message("trailing.comma.in.json"), pos); + } + pos++; + return global.wrapAsObject(arrayData); + default: + if (state == STATE_ELEMENT_PARSED) { + throw expectedError(pos, ", or ]", toString(c)); + } + final long index = arrayData.length(); + arrayData = arrayData.ensure(index).set((int) index, parseLiteral(), true); + state = STATE_ELEMENT_PARSED; break; } } - return result; + throw expectedError(pos, ", or ]", "eof"); } - /** - * Parse an object literal from the token stream - * @return the object literal as a Node - */ - private ObjectNode objectLiteral() { - // Capture LBRACE token. - final long objectToken = token; - // LBRACE tested in caller. - next(); + private String parseString() { + // String buffer is only instantiated if string contains escape sequences. + int start = ++pos; + StringBuilder sb = null; - // Prepare to accumulate elements. - final List elements = new ArrayList<>(); + while (pos < length) { + final int c = next(); + if (c <= 0x1f) { + // Characters < 0x1f are not allowed in JSON strings. + throw syntaxError(pos, "String contains control character"); - // Create a block for the object literal. -loop: - while (true) { - switch (type) { - case RBRACE: - next(); - break loop; - - case COMMARIGHT: - next(); - // check for trailing comma - not allowed in JSON - if (type == RBRACE) { - throw error(AbstractParser.message("trailing.comma.in.json", type.getNameOrType())); + } else if (c == '\\') { + if (sb == null) { + sb = new StringBuilder(pos - start + 16); } - break; + sb.append(source, start, pos - 1); + sb.append(parseEscapeSequence()); + start = pos; - default: - // Get and add the next property. - final PropertyNode property = propertyAssignment(); - elements.add(property); - - // Comma between property assigments is mandatory in JSON. - if (type != RBRACE && type != COMMARIGHT) { - throw error(AbstractParser.message("expected", ", or }", type.getNameOrType())); + } else if (c == '"') { + if (sb != null) { + sb.append(source, start, pos - 1); + return sb.toString(); } - break; + return source.substring(start, pos - 1); } } - // Construct new object literal. - return new ObjectNode(objectToken, finish, elements); + throw error(Lexer.message("missing.close.quote"), pos, length); } - /** - * Parse a property assignment from the token stream - * @return the property assignment as a Node - */ - private PropertyNode propertyAssignment() { - // Capture firstToken. - final long propertyToken = token; - LiteralNode name = null; - - if (type == STRING) { - name = getStringLiteral(); - } else if (type == ESCSTRING) { - name = getLiteral(); + private char parseEscapeSequence() { + final int c = next(); + switch (c) { + case '"': + return '"'; + case '\\': + return '\\'; + case '/': + return '/'; + case 'b': + return '\b'; + case 'f': + return '\f'; + case 'n': + return '\n'; + case 'r': + return '\r'; + case 't': + return '\t'; + case 'u': + return parseUnicodeEscape(); + default: + throw error(Lexer.message("invalid.escape.char"), pos - 1, length); } - - if (name != null) { - expect(COLON); - final Expression value = jsonLiteral(); - return new PropertyNode(propertyToken, value.getFinish(), name, value, null, null); - } - - // Raise an error. - throw error(AbstractParser.message("expected", "string", type.getNameOrType())); } + private char parseUnicodeEscape() { + return (char) (parseHexDigit() << 12 | parseHexDigit() << 8 | parseHexDigit() << 4 | parseHexDigit()); + } + + private int parseHexDigit() { + final int c = next(); + if (c >= '0' && c <= '9') { + return c - '0'; + } else if (c >= 'A' && c <= 'F') { + return c + 10 - 'A'; + } else if (c >= 'a' && c <= 'f') { + return c + 10 - 'a'; + } + throw error(Lexer.message("invalid.hex"), pos - 1, length); + } + + private boolean isDigit(final int c) { + return c >= '0' && c <= '9'; + } + + private void skipDigits() { + while (pos < length) { + final int c = peek(); + if (!isDigit(c)) { + break; + } + pos++; + } + } + + private Number parseNumber() { + final int start = pos; + int c = next(); + + if (c == '-') { + c = next(); + } + if (!isDigit(c)) { + throw numberError(start); + } + // no more digits allowed after 0 + if (c != '0') { + skipDigits(); + } + + // fraction + if (peek() == '.') { + pos++; + if (!isDigit(next())) { + throw numberError(pos - 1); + } + skipDigits(); + } + + // exponent + c = peek(); + if (c == 'e' || c == 'E') { + pos++; + c = next(); + if (c == '-' || c == '+') { + c = next(); + } + if (!isDigit(c)) { + throw numberError(pos - 1); + } + skipDigits(); + } + + final double d = Double.parseDouble(source.substring(start, pos)); + if (JSType.isRepresentableAsInt(d)) { + return (int) d; + } else if (JSType.isRepresentableAsLong(d)) { + return (long) d; + } + return d; + } + + private Object parseKeyword(final String keyword, final Object value) { + if (!source.regionMatches(pos, keyword, 0, keyword.length())) { + throw expectedError(pos, "json literal", "ident"); + } + pos += keyword.length(); + return value; + } + + private int peek() { + if (pos >= length) { + return -1; + } + return source.charAt(pos); + } + + private int next() { + final int next = peek(); + pos++; + return next; + } + + private void skipWhiteSpace() { + while (pos < length) { + switch (peek()) { + case '\t': + case '\r': + case '\n': + case ' ': + pos++; + break; + default: + return; + } + } + } + + private static String toString(final int c) { + return c == EOF ? "eof" : String.valueOf((char) c); + } + + ParserException error(final String message, final int start, final int length) throws ParserException { + final long token = Token.toDesc(STRING, start, length); + final int pos = Token.descPosition(token); + final Source src = Source.sourceFor("", source); + final int lineNum = src.getLine(pos); + final int columnNum = src.getColumn(pos); + final String formatted = ErrorManager.format(message, src, lineNum, columnNum, token); + return new ParserException(JSErrorType.SYNTAX_ERROR, formatted, src, lineNum, columnNum, token); + } + + private ParserException error(final String message, final int start) { + return error(message, start, length); + } + + private ParserException numberError(final int start) { + return error(Lexer.message("json.invalid.number"), start); + } + + private ParserException expectedError(final int start, final String expected, final String found) { + return error(AbstractParser.message("expected", expected, found), start); + } + + private ParserException syntaxError(final int start, final String reason) { + final String message = ECMAErrors.getMessage("syntax.error.invalid.json", reason); + return error(message, start); + } } diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/parser/Lexer.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/parser/Lexer.java index 04778cdbda0..5fe6ee80783 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/parser/Lexer.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/parser/Lexer.java @@ -93,9 +93,6 @@ public class Lexer extends Scanner { private static final String SPACETAB = " \t"; // ASCII space and tab private static final String LFCR = "\n\r"; // line feed and carriage return (ctrl-m) - private static final String JSON_WHITESPACE_EOL = LFCR; - private static final String JSON_WHITESPACE = SPACETAB + LFCR; - private static final String JAVASCRIPT_WHITESPACE_EOL = LFCR + "\u2028" + // line separator @@ -384,24 +381,6 @@ public class Lexer extends Scanner { return JAVASCRIPT_WHITESPACE_EOL.indexOf(ch) != -1; } - /** - * Test whether a char is valid JSON whitespace - * @param ch a char - * @return true if valid JSON whitespace - */ - public static boolean isJsonWhitespace(final char ch) { - return JSON_WHITESPACE.indexOf(ch) != -1; - } - - /** - * Test whether a char is valid JSON end of line - * @param ch a char - * @return true if valid JSON end of line - */ - public static boolean isJsonEOL(final char ch) { - return JSON_WHITESPACE_EOL.indexOf(ch) != -1; - } - /** * Test if char is a string delimiter, e.g. '\' or '"'. Also scans exec * strings ('`') in scripting mode. diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/JSONFunctions.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/JSONFunctions.java index a400bbd9adb..9622b59a17c 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/JSONFunctions.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/JSONFunctions.java @@ -25,19 +25,11 @@ package jdk.nashorn.internal.runtime; -import static jdk.nashorn.internal.runtime.Source.sourceFor; - import java.lang.invoke.MethodHandle; import java.util.Iterator; import java.util.concurrent.Callable; -import jdk.nashorn.internal.ir.LiteralNode; -import jdk.nashorn.internal.ir.Node; -import jdk.nashorn.internal.ir.ObjectNode; -import jdk.nashorn.internal.ir.PropertyNode; -import jdk.nashorn.internal.ir.UnaryNode; import jdk.nashorn.internal.objects.Global; import jdk.nashorn.internal.parser.JSONParser; -import jdk.nashorn.internal.parser.TokenType; import jdk.nashorn.internal.runtime.arrays.ArrayIndex; import jdk.nashorn.internal.runtime.linker.Bootstrap; @@ -78,20 +70,18 @@ public final class JSONFunctions { * @return Object representation of JSON text given */ public static Object parse(final Object text, final Object reviver) { - final String str = JSType.toString(text); - final JSONParser parser = new JSONParser(sourceFor("", str), new Context.ThrowErrorManager()); - - Node node; + final String str = JSType.toString(text); + final Global global = Context.getGlobal(); + final JSONParser parser = new JSONParser(str, global); + final Object value; try { - node = parser.parse(); + value = parser.parse(); } catch (final ParserException e) { throw ECMAErrors.syntaxError(e, "invalid.json", e.getMessage()); } - final Global global = Context.getGlobal(); - final Object unfiltered = convertNode(global, node); - return applyReviver(global, unfiltered, reviver); + return applyReviver(global, value, reviver); } // -- Internals only below this point @@ -137,61 +127,6 @@ public final class JSONFunctions { } } - // Converts IR node to runtime value - private static Object convertNode(final Global global, final Node node) { - if (node instanceof LiteralNode) { - // check for array literal - if (node.tokenType() == TokenType.ARRAY) { - assert node instanceof LiteralNode.ArrayLiteralNode; - final Node[] elements = ((LiteralNode.ArrayLiteralNode)node).getValue(); - - // NOTE: We cannot use LiteralNode.isNumericArray() here as that - // method uses symbols of element nodes. Since we don't do lower - // pass, there won't be any symbols! - if (isNumericArray(elements)) { - final double[] values = new double[elements.length]; - int index = 0; - - for (final Node elem : elements) { - values[index++] = JSType.toNumber(convertNode(global, elem)); - } - return global.wrapAsObject(values); - } - - final Object[] values = new Object[elements.length]; - int index = 0; - - for (final Node elem : elements) { - values[index++] = convertNode(global, elem); - } - - return global.wrapAsObject(values); - } - - return ((LiteralNode)node).getValue(); - - } else if (node instanceof ObjectNode) { - final ObjectNode objNode = (ObjectNode) node; - final ScriptObject object = global.newObject(); - - for (final PropertyNode pNode: objNode.getElements()) { - final Node valueNode = pNode.getValue(); - - final String name = pNode.getKeyName(); - final Object value = convertNode(global, valueNode); - setPropertyValue(object, name, value); - } - - return object; - } else if (node instanceof UnaryNode) { - // UnaryNode used only to represent negative number JSON value - final UnaryNode unaryNode = (UnaryNode)node; - return -((LiteralNode)unaryNode.getExpression()).getNumber(); - } else { - return null; - } - } - // add a new property if does not exist already, or else set old property private static void setPropertyValue(final ScriptObject sobj, final String name, final Object value) { final int index = ArrayIndex.getArrayIndex(name); @@ -207,14 +142,4 @@ public final class JSONFunctions { } } - // does the given IR node represent a numeric array? - private static boolean isNumericArray(final Node[] values) { - for (final Node node : values) { - if (node instanceof LiteralNode && ((LiteralNode)node).getValue() instanceof Number) { - continue; - } - return false; - } - return true; - } } diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/PropertyMap.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/PropertyMap.java index 30a151322e6..2014c643796 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/PropertyMap.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/PropertyMap.java @@ -486,7 +486,7 @@ public final class PropertyMap implements Iterable, Serializable { * * @return New {@link PropertyMap} with {@link Property} replaced. */ - PropertyMap replaceProperty(final Property oldProperty, final Property newProperty) { + public PropertyMap replaceProperty(final Property oldProperty, final Property newProperty) { if (listeners != null) { listeners.propertyModified(oldProperty, newProperty); } diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/scripts/JO.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/scripts/JO.java index e5346cb2c07..2388a9781bd 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/scripts/JO.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/scripts/JO.java @@ -25,7 +25,6 @@ package jdk.nashorn.internal.scripts; -import jdk.nashorn.internal.codegen.SpillObjectCreator; import jdk.nashorn.internal.runtime.PropertyMap; import jdk.nashorn.internal.runtime.ScriptObject; @@ -64,8 +63,9 @@ public class JO extends ScriptObject { } /** - * Constructor that takes a pre-initialized spill pool. Used for - * by {@link SpillObjectCreator} for intializing object literals + * Constructor that takes a pre-initialized spill pool. Used by + * {@link jdk.nashorn.internal.codegen.SpillObjectCreator} and + * {@link jdk.nashorn.internal.parser.JSONParser} for initializing object literals * * @param map property map * @param primitiveSpill primitive spill pool diff --git a/nashorn/test/examples/json-parser-micro.js b/nashorn/test/examples/json-parser-micro.js new file mode 100644 index 00000000000..619c35f4822 --- /dev/null +++ b/nashorn/test/examples/json-parser-micro.js @@ -0,0 +1,315 @@ +/* + * Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved. + * + * Redistribution and use in source and binary forms, with or without + * modification, are permitted provided that the following conditions + * are met: + * + * - Redistributions of source code must retain the above copyright + * notice, this list of conditions and the following disclaimer. + * + * - Redistributions in binary form must reproduce the above copyright + * notice, this list of conditions and the following disclaimer in the + * documentation and/or other materials provided with the distribution. + * + * - Neither the name of Oracle nor the names of its + * contributors may be used to endorse or promote products derived + * from this software without specific prior written permission. + * + * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS + * IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, + * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR + * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR + * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, + * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, + * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR + * PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF + * LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING + * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS + * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + */ + +function bench() { + var start = Date.now(); + for (var i = 0; i < 2000; i++) { + JSON.parse(String(json)); + } + print("1000 iterations in", Date.now() - start, "millis"); +} + +var json = '[\ + {\ + "_id": "54ca34171d3ade49782294c8",\ + "index": 0,\ + "guid": "ed0e74d5-ac63-47b6-8938-1750abab5770",\ + "isActive": false,\ + "balance": "$1,996.19",\ + "picture": "http://placehold.it/32x32",\ + "age": 39,\ + "eyeColor": "green",\ + "name": "Rose Graham",\ + "gender": "male",\ + "company": "PRIMORDIA",\ + "email": "rosegraham@primordia.com",\ + "phone": "+1 (985) 600-3551",\ + "address": "364 Melba Court, Succasunna, Texas, 8393",\ + "about": "Sunt commodo cillum occaecat velit eu eiusmod ex eiusmod sunt deserunt nulla proident incididunt. Incididunt ullamco Lorem elit do culpa esse do ex dolor aliquip labore. Ullamco velit laboris incididunt dolor. Nostrud dolor sint pariatur fugiat ullamco exercitation. Eu laboris do cupidatat eiusmod incididunt mollit occaecat voluptate.",\ + "registered": "2014-03-13T12:05:14 -01:00",\ + "latitude": 18.55665,\ + "longitude": 81.641001,\ + "tags": [\ + "sint",\ + "Lorem",\ + "veniam",\ + "quis",\ + "proident",\ + "consectetur",\ + "consequat"\ + ],\ + "friends": [\ + {\ + "id": 0,\ + "name": "Evangelina Morgan"\ + },\ + {\ + "id": 1,\ + "name": "Saunders Snyder"\ + },\ + {\ + "id": 2,\ + "name": "Walker Wood"\ + }\ + ],\ + "greeting": "Hello, Rose Graham! You have 1 unread messages.",\ + "favoriteFruit": "strawberry"\ + },\ + {\ + "_id": "54ca34176790c4c60fcae085",\ + "index": 1,\ + "guid": "9dc42e4c-b58f-4d92-a2ee-968d2b627d92",\ + "isActive": true,\ + "balance": "$3,832.97",\ + "picture": "http://placehold.it/32x32",\ + "age": 40,\ + "eyeColor": "brown",\ + "name": "Delaney Cherry",\ + "gender": "male",\ + "company": "INJOY",\ + "email": "delaneycherry@injoy.com",\ + "phone": "+1 (807) 463-2295",\ + "address": "470 Hale Avenue, Mulberry, District Of Columbia, 5455",\ + "about": "Deserunt sit cupidatat elit Lorem excepteur ex. Magna officia minim cupidatat nulla enim deserunt. Amet ex in tempor commodo consequat non ad qui elit cupidatat esse labore sint.",\ + "registered": "2014-03-27T23:06:33 -01:00",\ + "latitude": -4.984238,\ + "longitude": 116.039285,\ + "tags": [\ + "minim",\ + "velit",\ + "aute",\ + "minim",\ + "id",\ + "enim",\ + "enim"\ + ],\ + "friends": [\ + {\ + "id": 0,\ + "name": "Barrera Flowers"\ + },\ + {\ + "id": 1,\ + "name": "Leann Larson"\ + },\ + {\ + "id": 2,\ + "name": "Latoya Petty"\ + }\ + ],\ + "greeting": "Hello, Delaney Cherry! You have 2 unread messages.",\ + "favoriteFruit": "strawberry"\ + },\ + {\ + "_id": "54ca3417920666f00c54bfc4",\ + "index": 2,\ + "guid": "f91e08f8-1598-49bc-a08b-bb48f0cc1751",\ + "isActive": true,\ + "balance": "$2,932.84",\ + "picture": "http://placehold.it/32x32",\ + "age": 28,\ + "eyeColor": "brown",\ + "name": "Mosley Hammond",\ + "gender": "male",\ + "company": "AQUACINE",\ + "email": "mosleyhammond@aquacine.com",\ + "phone": "+1 (836) 598-2591",\ + "address": "879 Columbia Place, Seymour, Montana, 4897",\ + "about": "Sunt laborum incididunt et elit in deserunt deserunt irure enim ea qui non. Minim nisi sint aute veniam reprehenderit veniam reprehenderit. Elit enim eu voluptate eu cupidatat nulla ea incididunt exercitation voluptate ut aliquip excepteur ipsum. Consequat anim fugiat irure Lorem anim consectetur est.",\ + "registered": "2014-07-27T05:05:58 -02:00",\ + "latitude": -43.608015,\ + "longitude": -38.33894,\ + "tags": [\ + "proident",\ + "incididunt",\ + "eiusmod",\ + "anim",\ + "consectetur",\ + "qui",\ + "excepteur"\ + ],\ + "friends": [\ + {\ + "id": 0,\ + "name": "Hanson Davidson"\ + },\ + {\ + "id": 1,\ + "name": "Autumn Kaufman"\ + },\ + {\ + "id": 2,\ + "name": "Tammy Foley"\ + }\ + ],\ + "greeting": "Hello, Mosley Hammond! You have 4 unread messages.",\ + "favoriteFruit": "apple"\ + },\ + {\ + "_id": "54ca341753b67572a2b04935",\ + "index": 3,\ + "guid": "3377416b-43a2-4f9e-ada3-2479e13b44b8",\ + "isActive": false,\ + "balance": "$3,821.54",\ + "picture": "http://placehold.it/32x32",\ + "age": 31,\ + "eyeColor": "green",\ + "name": "Mueller Barrett",\ + "gender": "male",\ + "company": "GROK",\ + "email": "muellerbarrett@grok.com",\ + "phone": "+1 (890) 535-2834",\ + "address": "571 Norwood Avenue, Westwood, Arkansas, 2164",\ + "about": "Occaecat est sunt commodo ut ex excepteur elit nulla velit minim commodo commodo esse. Lorem quis eu minim consectetur. Cupidatat cupidatat consequat sit eu ex non quis nulla veniam sint enim excepteur. Consequat minim duis do do minim fugiat minim elit laborum ut velit. Occaecat laboris veniam sint reprehenderit.",\ + "registered": "2014-07-18T17:15:35 -02:00",\ + "latitude": 10.746577,\ + "longitude": -160.266041,\ + "tags": [\ + "reprehenderit",\ + "veniam",\ + "sint",\ + "commodo",\ + "exercitation",\ + "cillum",\ + "sunt"\ + ],\ + "friends": [\ + {\ + "id": 0,\ + "name": "Summers Finch"\ + },\ + {\ + "id": 1,\ + "name": "Tracie Mcdaniel"\ + },\ + {\ + "id": 2,\ + "name": "Ayers Patrick"\ + }\ + ],\ + "greeting": "Hello, Mueller Barrett! You have 7 unread messages.",\ + "favoriteFruit": "apple"\ + },\ + {\ + "_id": "54ca34172775ab9615db0d1d",\ + "index": 4,\ + "guid": "a3102a3e-3f08-4df3-b5b5-62eff985d5ca",\ + "isActive": true,\ + "balance": "$3,962.27",\ + "picture": "http://placehold.it/32x32",\ + "age": 34,\ + "eyeColor": "green",\ + "name": "Patrick Foster",\ + "gender": "male",\ + "company": "QUAREX",\ + "email": "patrickfoster@quarex.com",\ + "phone": "+1 (805) 577-2362",\ + "address": "640 Richards Street, Roberts, American Samoa, 5530",\ + "about": "Aute occaecat occaecat ad eiusmod esse aliqua ullamco minim. Exercitation aute ut ex nostrud deserunt laboris officia amet enim do. Cillum officia laborum occaecat eiusmod reprehenderit ex et aliqua minim elit ex aliqua mollit. Occaecat dolor in fugiat laboris aliquip nisi ad voluptate duis eiusmod ad do.",\ + "registered": "2014-07-22T16:45:35 -02:00",\ + "latitude": 6.609025,\ + "longitude": -5.357026,\ + "tags": [\ + "ea",\ + "ut",\ + "excepteur",\ + "enim",\ + "ad",\ + "non",\ + "sit"\ + ],\ + "friends": [\ + {\ + "id": 0,\ + "name": "Duncan Lewis"\ + },\ + {\ + "id": 1,\ + "name": "Alyce Benton"\ + },\ + {\ + "id": 2,\ + "name": "Angelique Larsen"\ + }\ + ],\ + "greeting": "Hello, Patrick Foster! You have 1 unread messages.",\ + "favoriteFruit": "strawberry"\ + },\ + {\ + "_id": "54ca3417a190f26fef815f6d",\ + "index": 5,\ + "guid": "c09663dd-bb0e-45a4-960c-232c0e8a9486",\ + "isActive": false,\ + "balance": "$1,871.12",\ + "picture": "http://placehold.it/32x32",\ + "age": 20,\ + "eyeColor": "blue",\ + "name": "Foreman Chaney",\ + "gender": "male",\ + "company": "DEMINIMUM",\ + "email": "foremanchaney@deminimum.com",\ + "phone": "+1 (966) 523-2182",\ + "address": "960 Granite Street, Sunnyside, Tennessee, 1097",\ + "about": "Adipisicing nisi qui id sit incididunt aute exercitation veniam consequat ipsum sit irure. Aute officia commodo Lorem consequat. Labore exercitation consequat voluptate deserunt consequat do est fugiat nisi eu dolor minim id ea.",\ + "registered": "2015-01-21T00:18:00 -01:00",\ + "latitude": -69.841726,\ + "longitude": 121.809383,\ + "tags": [\ + "laboris",\ + "sunt",\ + "exercitation",\ + "enim",\ + "anim",\ + "excepteur",\ + "tempor"\ + ],\ + "friends": [\ + {\ + "id": 0,\ + "name": "Espinoza Johnston"\ + },\ + {\ + "id": 1,\ + "name": "Doreen Holder"\ + },\ + {\ + "id": 2,\ + "name": "William Ellison"\ + }\ + ],\ + "greeting": "Hello, Foreman Chaney! You have 5 unread messages.",\ + "favoriteFruit": "strawberry"\ + }\ +]'; + +for (var i = 0; i < 100; i++) { + bench(); +} diff --git a/nashorn/test/script/basic/JDK-8062141.js b/nashorn/test/script/basic/JDK-8062141.js new file mode 100644 index 00000000000..953d92f90dd --- /dev/null +++ b/nashorn/test/script/basic/JDK-8062141.js @@ -0,0 +1,91 @@ +/* + * Copyright (c) 2015, 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. + */ + +/** + * JDK-8062141: Various performance issues parsing JSON + * + * @test + * @run + */ + +function testJson(json) { + try { + print(JSON.stringify(JSON.parse(json))); + } catch (error) { + print(error); + } +} + +testJson('"\\u003f"'); +testJson('"\\u0"'); +testJson('"\\u0"'); +testJson('"\\u00"'); +testJson('"\\u003"'); +testJson('"\\u003x"'); +testJson('"\\"'); +testJson('"'); +testJson('+1'); +testJson('-1'); +testJson('1.'); +testJson('.1'); +testJson('01'); +testJson('1e'); +testJson('1e0'); +testJson('1a'); +testJson('1e+'); +testJson('1e-'); +testJson('0.0e+0'); +testJson('0.0e-0'); +testJson('[]'); +testJson('[ 1 ]'); +testJson('[1,]'); +testJson('[ 1 , 2 ]'); +testJson('[1, 2'); +testJson('{}'); +testJson('{ "a" : "b" }'); +testJson('{ "a" : "b" '); +testJson('{ "a" : }'); +testJson('true'); +testJson('tru'); +testJson('true1'); +testJson('false'); +testJson('fals'); +testJson('falser'); +testJson('null'); +testJson('nul'); +testJson('null0'); +testJson('{} 0'); +testJson('{} a'); +testJson('[] 0'); +testJson('[] a'); +testJson('1 0'); +testJson('1 a'); +testJson('["a":true]'); +testJson('{"a",truer}'); +testJson('{"a":truer}'); +testJson('[1, 2, 3]'); +testJson('[9223372036854774000, 9223372036854775000, 9223372036854776000]'); +testJson('[1.1, 1.2, 1.3]'); +testJson('[1, 1.2, 9223372036854776000, null, true]'); +testJson('{ "a" : "string" , "b": 1 , "c" : 1.2 , "d" : 9223372036854776000 , "e" : null , "f" : true }'); + diff --git a/nashorn/test/script/basic/JDK-8062141.js.EXPECTED b/nashorn/test/script/basic/JDK-8062141.js.EXPECTED new file mode 100644 index 00000000000..18dd8f4092f --- /dev/null +++ b/nashorn/test/script/basic/JDK-8062141.js.EXPECTED @@ -0,0 +1,120 @@ +"?" +SyntaxError: Invalid JSON: :1:4 Invalid hex digit +"\u0" + ^ +SyntaxError: Invalid JSON: :1:4 Invalid hex digit +"\u0" + ^ +SyntaxError: Invalid JSON: :1:5 Invalid hex digit +"\u00" + ^ +SyntaxError: Invalid JSON: :1:6 Invalid hex digit +"\u003" + ^ +SyntaxError: Invalid JSON: :1:6 Invalid hex digit +"\u003x" + ^ +SyntaxError: Invalid JSON: :1:3 Missing close quote +"\" + ^ +SyntaxError: Invalid JSON: :1:1 Missing close quote +" + ^ +SyntaxError: Invalid JSON: :1:0 Expected json literal but found + ++1 +^ +-1 +SyntaxError: Invalid JSON: :1:2 Invalid JSON number format +1. + ^ +SyntaxError: Invalid JSON: :1:0 Invalid JSON number format +.1 +^ +SyntaxError: Invalid JSON: :1:1 Expected eof but found 1 +01 + ^ +SyntaxError: Invalid JSON: :1:2 Invalid JSON number format +1e + ^ +1 +SyntaxError: Invalid JSON: :1:1 Expected eof but found a +1a + ^ +SyntaxError: Invalid JSON: :1:3 Invalid JSON number format +1e+ + ^ +SyntaxError: Invalid JSON: :1:3 Invalid JSON number format +1e- + ^ +0 +0 +[] +[1] +SyntaxError: Invalid JSON: :1:3 Trailing comma is not allowed in JSON +[1,] + ^ +[1,2] +SyntaxError: Invalid JSON: :1:5 Expected , or ] but found eof +[1, 2 + ^ +{} +{"a":"b"} +SyntaxError: Invalid JSON: :1:12 Expected , or } but found eof +{ "a" : "b" + ^ +SyntaxError: Invalid JSON: :1:8 Expected json literal but found } +{ "a" : } + ^ +true +SyntaxError: Invalid JSON: :1:0 Expected json literal but found ident +tru +^ +SyntaxError: Invalid JSON: :1:4 Expected eof but found 1 +true1 + ^ +false +SyntaxError: Invalid JSON: :1:0 Expected json literal but found ident +fals +^ +SyntaxError: Invalid JSON: :1:5 Expected eof but found r +falser + ^ +null +SyntaxError: Invalid JSON: :1:0 Expected json literal but found ident +nul +^ +SyntaxError: Invalid JSON: :1:4 Expected eof but found 0 +null0 + ^ +SyntaxError: Invalid JSON: :1:3 Expected eof but found 0 +{} 0 + ^ +SyntaxError: Invalid JSON: :1:3 Expected eof but found a +{} a + ^ +SyntaxError: Invalid JSON: :1:3 Expected eof but found 0 +[] 0 + ^ +SyntaxError: Invalid JSON: :1:3 Expected eof but found a +[] a + ^ +SyntaxError: Invalid JSON: :1:2 Expected eof but found 0 +1 0 + ^ +SyntaxError: Invalid JSON: :1:2 Expected eof but found a +1 a + ^ +SyntaxError: Invalid JSON: :1:4 Expected , or ] but found : +["a":true] + ^ +SyntaxError: Invalid JSON: :1:4 Expected : but found , +{"a",truer} + ^ +SyntaxError: Invalid JSON: :1:9 Expected , or } but found r +{"a":truer} + ^ +[1,2,3] +[9223372036854773800,9223372036854774800,9223372036854776000] +[1.1,1.2,1.3] +[1,1.2,9223372036854776000,null,true] +{"a":"string","b":1,"c":1.2,"d":9223372036854776000,"e":null,"f":true} diff --git a/nashorn/test/script/basic/NASHORN-623.js.EXPECTED b/nashorn/test/script/basic/NASHORN-623.js.EXPECTED index 5755489b2c9..6b1f05bdf19 100644 --- a/nashorn/test/script/basic/NASHORN-623.js.EXPECTED +++ b/nashorn/test/script/basic/NASHORN-623.js.EXPECTED @@ -1,3 +1,3 @@ -SyntaxError: Invalid JSON: :1:12 Expected number but found ident +SyntaxError: Invalid JSON: :1:11 Invalid JSON number format { "test" : -xxx } - ^ + ^ From 0e498bf1aab53623f7e450dd1cd928e1a73dc548 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Hannes=20Walln=C3=B6fer?= Date: Thu, 5 Feb 2015 14:47:28 +0100 Subject: [PATCH 51/58] 8068872: Nashorn JSON.parse drops numeric keys Reviewed-by: attila, lagergren --- .../nashorn/internal/objects/NativeArray.java | 3 -- .../internal/runtime/ScriptObject.java | 8 +++-- .../internal/runtime/arrays/ArrayData.java | 6 ++-- .../runtime/arrays/ContinuousArrayData.java | 2 +- .../internal/runtime/arrays/IntArrayData.java | 4 ++- .../runtime/arrays/LongArrayData.java | 4 ++- .../runtime/arrays/NumberArrayData.java | 4 ++- .../runtime/arrays/ObjectArrayData.java | 4 ++- .../runtime/arrays/SparseArrayData.java | 9 ++++- nashorn/test/script/basic/JDK-8068872.js | 34 +++++++++++++++++++ .../test/script/basic/JDK-8068872.js.EXPECTED | 4 +++ 11 files changed, 69 insertions(+), 13 deletions(-) create mode 100644 nashorn/test/script/basic/JDK-8068872.js create mode 100644 nashorn/test/script/basic/JDK-8068872.js.EXPECTED diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/objects/NativeArray.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/objects/NativeArray.java index 0033ed71915..f9ab01cc70f 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/objects/NativeArray.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/objects/NativeArray.java @@ -120,9 +120,6 @@ public final class NativeArray extends ScriptObject implements OptimisticBuiltin this(ArrayData.allocate(array.length)); ArrayData arrayData = this.getArray(); - if (array.length > 0) { - arrayData.ensure(array.length - 1); - } for (int index = 0; index < array.length; index++) { final Object value = array[index]; diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/ScriptObject.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/ScriptObject.java index c3eca18fee1..358cc05c960 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/ScriptObject.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/ScriptObject.java @@ -722,8 +722,12 @@ public abstract class ScriptObject implements PropertyAccess, Cloneable { public void defineOwnProperty(final int index, final Object value) { assert isValidArrayIndex(index) : "invalid array index"; final long longIndex = ArrayIndex.toLongIndex(index); - doesNotHaveEnsureDelete(longIndex, getArray().length(), false); - setArray(getArray().ensure(longIndex).set(index,value, false)); + final long oldLength = getArray().length(); + if (longIndex >= oldLength) { + setArray(getArray().ensure(longIndex)); + doesNotHaveEnsureDelete(longIndex, oldLength, false); + } + setArray(getArray().set(index,value, false)); } private void checkIntegerKey(final String key) { diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/ArrayData.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/ArrayData.java index 6c98028a602..db871956b30 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/ArrayData.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/ArrayData.java @@ -118,7 +118,7 @@ public abstract class ArrayData { return new SparseArrayData(this, safeIndex + 1); } //known to fit in int - return toRealArrayData((int)safeIndex).ensure(safeIndex); + return toRealArrayData((int)safeIndex); } return this; } @@ -497,7 +497,9 @@ public abstract class ArrayData { public abstract ArrayData shiftRight(final int by); /** - * Ensure that the given index exists and won't fail subsequent + * Ensure that the given index exists and won't fail in a subsequent access. + * If {@code safeIndex} is equal or greater than the current length the length is + * updated to {@code safeIndex + 1}. * * @param safeIndex the index to ensure wont go out of bounds * @return new array data (or same) diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/ContinuousArrayData.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/ContinuousArrayData.java index 2fdd184e45c..e892f9fc139 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/ContinuousArrayData.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/ContinuousArrayData.java @@ -57,7 +57,7 @@ public abstract class ContinuousArrayData extends ArrayData { } /** - * Check if we can put one more element at the end of this continous + * Check if we can put one more element at the end of this continuous * array without reallocating, or if we are overwriting an already * allocated element * diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/IntArrayData.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/IntArrayData.java index a70dec2747c..73a50dc3e29 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/IntArrayData.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/IntArrayData.java @@ -221,7 +221,9 @@ final class IntArrayData extends ContinuousArrayData implements IntElements { final int newLength = ArrayData.nextSize((int)safeIndex); array = Arrays.copyOf(array, newLength); } - setLength(safeIndex + 1); + if (safeIndex >= length()) { + setLength(safeIndex + 1); + } return this; } diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/LongArrayData.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/LongArrayData.java index bf3431926f0..da26eecf37d 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/LongArrayData.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/LongArrayData.java @@ -157,7 +157,9 @@ final class LongArrayData extends ContinuousArrayData implements IntOrLongElemen final int newLength = ArrayData.nextSize((int)safeIndex); array = Arrays.copyOf(array, newLength); } - setLength(safeIndex + 1); + if (safeIndex >= length()) { + setLength(safeIndex + 1); + } return this; } diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/NumberArrayData.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/NumberArrayData.java index d938d1d02d7..0f633a9225c 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/NumberArrayData.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/NumberArrayData.java @@ -139,7 +139,9 @@ final class NumberArrayData extends ContinuousArrayData implements NumericElemen final int newLength = ArrayData.nextSize((int)safeIndex); array = Arrays.copyOf(array, newLength); //todo fill with nan or never accessed? } - setLength(safeIndex + 1); + if (safeIndex >= length()) { + setLength(safeIndex + 1); + } return this; } diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/ObjectArrayData.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/ObjectArrayData.java index ebaa3d6d030..3ccf9933530 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/ObjectArrayData.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/ObjectArrayData.java @@ -123,7 +123,9 @@ final class ObjectArrayData extends ContinuousArrayData implements AnyElements { final int newLength = ArrayData.nextSize((int)safeIndex); array = Arrays.copyOf(array, newLength); //fill with undefined or OK? TODO } - setLength(safeIndex + 1); + if (safeIndex >= length()) { + setLength(safeIndex + 1); + } return this; } diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/SparseArrayData.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/SparseArrayData.java index ff569d449e6..ede8d877608 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/SparseArrayData.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/runtime/arrays/SparseArrayData.java @@ -135,10 +135,17 @@ class SparseArrayData extends ArrayData { @Override public ArrayData ensure(final long safeIndex) { + // Usually #ensure only needs to be called if safeIndex is greater or equal current length. + // SparseArrayData is an exception as an index smaller than our current length may still + // exceed the underlying ArrayData's capacity. Because of this, SparseArrayData invokes + // its ensure method internally in various places where other ArrayData subclasses don't, + // making it safe for outside uses to only call ensure(safeIndex) if safeIndex >= length. if (safeIndex < maxDenseLength && underlying.length() <= safeIndex) { underlying = underlying.ensure(safeIndex); } - setLength(Math.max(safeIndex + 1, length())); + if (safeIndex >= length()) { + setLength(safeIndex + 1); + } return this; } diff --git a/nashorn/test/script/basic/JDK-8068872.js b/nashorn/test/script/basic/JDK-8068872.js new file mode 100644 index 00000000000..4d25ccf90a8 --- /dev/null +++ b/nashorn/test/script/basic/JDK-8068872.js @@ -0,0 +1,34 @@ +/* + * Copyright (c) 2015, 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. + */ + +/** + * JDK-8068872: Nashorn JSON.parse drops numeric keys + * + * @test + * @run + */ + +print(JSON.stringify(JSON.parse('{"3": 1, "5": "a"}'))); +print(JSON.stringify(JSON.parse('{"5": 1, "3": "a"}'))); +print(JSON.stringify(JSON.parse('{"0": 1, "4294967294": "a"}'))); +print(JSON.stringify(JSON.parse('{"4294967294": 1, "0": "a"}'))); diff --git a/nashorn/test/script/basic/JDK-8068872.js.EXPECTED b/nashorn/test/script/basic/JDK-8068872.js.EXPECTED new file mode 100644 index 00000000000..a191d207b15 --- /dev/null +++ b/nashorn/test/script/basic/JDK-8068872.js.EXPECTED @@ -0,0 +1,4 @@ +{"3":1,"5":"a"} +{"3":"a","5":1} +{"0":1,"4294967294":"a"} +{"0":"a","4294967294":1} From 3f9c2f13ee54166e44f6d354c339a0287c2b7e66 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Hannes=20Walln=C3=B6fer?= Date: Thu, 5 Feb 2015 16:26:36 +0100 Subject: [PATCH 52/58] 8072626: Test for JDK-8068872 fails in tip Reviewed-by: lagergren, jlaskey --- .../classes/jdk/nashorn/internal/parser/JSONParser.java | 7 ++++--- 1 file changed, 4 insertions(+), 3 deletions(-) diff --git a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/parser/JSONParser.java b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/parser/JSONParser.java index e11a9efac71..f1f46ea6fad 100644 --- a/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/parser/JSONParser.java +++ b/nashorn/src/jdk.scripting.nashorn/share/classes/jdk/nashorn/internal/parser/JSONParser.java @@ -232,12 +232,13 @@ public class JSONParser { final long oldLength = arrayData.length(); final long longIndex = ArrayIndex.toLongIndex(index); ArrayData newArrayData = arrayData; - if (longIndex > oldLength) { - if (arrayData.canDelete(oldLength, longIndex - 1, false)) { + if (longIndex >= oldLength) { + newArrayData = newArrayData.ensure(longIndex); + if (longIndex > oldLength) { newArrayData = newArrayData.delete(oldLength, longIndex - 1); } } - return newArrayData.ensure(longIndex).set(index, value, false); + return newArrayData.set(index, value, false); } private static PropertyMap addObjectProperty(final PropertyMap propertyMap, final List values, From 5184f0183be4bb002fc6ec013806d7b91a685e42 Mon Sep 17 00:00:00 2001 From: David Katleman Date: Thu, 5 Feb 2015 11:43:28 -0800 Subject: [PATCH 53/58] Added tag jdk9-b49 for changeset dd613dc718a1 --- .hgtags-top-repo | 1 + 1 file changed, 1 insertion(+) diff --git a/.hgtags-top-repo b/.hgtags-top-repo index 26c2321b403..7bbd63e09a8 100644 --- a/.hgtags-top-repo +++ b/.hgtags-top-repo @@ -291,3 +291,4 @@ f7c11da0b0481d49cc7a65a453336c108191e821 jdk9-b42 12f1e276447bcc81516e85367d53e4f08897049d jdk9-b46 b6cca3e6175a69f39e5799b7349ddb0176630291 jdk9-b47 0064e246d83f6f9fc245c19b6d05041ecaf4b6d4 jdk9-b48 +d91ed1951b948210590ce1394bea5515357246ba jdk9-b49 From e58a6b0ce9a0875cbace0fb5312fe1fce40806c2 Mon Sep 17 00:00:00 2001 From: David Katleman Date: Thu, 5 Feb 2015 11:43:29 -0800 Subject: [PATCH 54/58] Added tag jdk9-b49 for changeset 6a22230c068f --- hotspot/.hgtags | 1 + 1 file changed, 1 insertion(+) diff --git a/hotspot/.hgtags b/hotspot/.hgtags index a0e96eab962..08b2b023cfe 100644 --- a/hotspot/.hgtags +++ b/hotspot/.hgtags @@ -451,3 +451,4 @@ c363a8b87e477ee45d6d3cb2a36cb365141bc596 jdk9-b38 a184ee1d717297bd35b7c3e35393e137921a3ed2 jdk9-b46 3b241fb72b8925b75941d612db762a6d5da66d02 jdk9-b47 cc775a4a24c7f5d9e624b4205e9fbd48a17331f6 jdk9-b48 +360cd1fc42f10941a9fd17cc32d5b85a22d12a0b jdk9-b49 From 00fa5a044f21fed3184c6619ff104e5d521872f4 Mon Sep 17 00:00:00 2001 From: David Katleman Date: Thu, 5 Feb 2015 11:43:29 -0800 Subject: [PATCH 55/58] Added tag jdk9-b49 for changeset 08067f67dde0 --- corba/.hgtags | 1 + 1 file changed, 1 insertion(+) diff --git a/corba/.hgtags b/corba/.hgtags index 5fa0c422fef..171c3db868c 100644 --- a/corba/.hgtags +++ b/corba/.hgtags @@ -291,3 +291,4 @@ e27c725d6c9d155667b35255f442d4ceb8c3c084 jdk9-b40 326f2068b4a4c05e2fa27d6acf93eba7b54b090d jdk9-b46 ee8447ca632e1d39180b4767c749db101bff7314 jdk9-b47 a13c49c5f2899b702652a460ed7aa73123e671e6 jdk9-b48 +9285d14eb7b6b0815679bae98dd936dbc136218d jdk9-b49 From 6680f977945e93b67eb10d4263c2b6f8fd1e4732 Mon Sep 17 00:00:00 2001 From: David Katleman Date: Thu, 5 Feb 2015 11:43:32 -0800 Subject: [PATCH 56/58] Added tag jdk9-b49 for changeset f9e5640d832e --- jaxp/.hgtags | 1 + 1 file changed, 1 insertion(+) diff --git a/jaxp/.hgtags b/jaxp/.hgtags index 664424cb343..2e3d8c48f55 100644 --- a/jaxp/.hgtags +++ b/jaxp/.hgtags @@ -291,3 +291,4 @@ a12d347f84176200593999f4da91ae2bb86865b2 jdk9-b39 74eaf7ad986576c792df4dbff05eed63e5727695 jdk9-b46 e391de88e69b59d7c618387e3cf91032f6991ce9 jdk9-b47 833051855168a973780fafeb6fc59e7370bcf400 jdk9-b48 +786058752e0ac3e48d7aef79e0885d29d6a2a7eb jdk9-b49 From 0ea4df7b6c1eadbdb0fd4bb1e21247d3e4c6b018 Mon Sep 17 00:00:00 2001 From: David Katleman Date: Thu, 5 Feb 2015 11:43:33 -0800 Subject: [PATCH 57/58] Added tag jdk9-b49 for changeset 6e8154707178 --- jdk/.hgtags | 1 + 1 file changed, 1 insertion(+) diff --git a/jdk/.hgtags b/jdk/.hgtags index c5a3e91f84e..4977aaab0a3 100644 --- a/jdk/.hgtags +++ b/jdk/.hgtags @@ -291,3 +291,4 @@ e336cbd8b15e959e70ed02f0f5e93fa76ebd4c07 jdk9-b41 efedac7f44ed41cea2b1038138047271f55aacba jdk9-b46 b641c14730ac05d9ec8b4f66e6fca3dc21adb403 jdk9-b47 ebb2eb7f1aec78eb6d8cc4c96f018afa11093cde jdk9-b48 +541a8cef4e0d54c3e4b52a98c6af3c31e2096669 jdk9-b49 From 125aff9d1987240bbfa0e489cd6ce62821c984a6 Mon Sep 17 00:00:00 2001 From: David Katleman Date: Thu, 5 Feb 2015 11:43:38 -0800 Subject: [PATCH 58/58] Added tag jdk9-b49 for changeset bdc353778a28 --- nashorn/.hgtags | 1 + 1 file changed, 1 insertion(+) diff --git a/nashorn/.hgtags b/nashorn/.hgtags index 76c29cdab83..ff3d2a9e158 100644 --- a/nashorn/.hgtags +++ b/nashorn/.hgtags @@ -282,3 +282,4 @@ dd7bbdf81a537106cfa9227d1a9a57849cb26b4d jdk9-b37 2ecf0a617f0f9af1ffd278a0c70e76f1946ce773 jdk9-b46 29046d42a95e5b9f105ab086a628bbd7f81c915d jdk9-b47 f08660f30051ba0b38ad00e692979b37d107c9c4 jdk9-b48 +2ae58b5f05f803a469f0f6c1ed72c6b5313f4ff0 jdk9-b49