Mon, 03 Mar 2014 15:23:01 +0100
8035948: Redesign property listeners for shared classes
Reviewed-by: sundar, lagergren
jlaskey@3 | 1 | /* |
jlaskey@7 | 2 | * Copyright (c) 2010, 2013, Oracle and/or its affiliates. All rights reserved. |
jlaskey@3 | 3 | * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. |
jlaskey@3 | 4 | * |
jlaskey@3 | 5 | * This code is free software; you can redistribute it and/or modify it |
jlaskey@3 | 6 | * under the terms of the GNU General Public License version 2 only, as |
jlaskey@3 | 7 | * published by the Free Software Foundation. Oracle designates this |
jlaskey@3 | 8 | * particular file as subject to the "Classpath" exception as provided |
jlaskey@3 | 9 | * by Oracle in the LICENSE file that accompanied this code. |
jlaskey@3 | 10 | * |
jlaskey@3 | 11 | * This code is distributed in the hope that it will be useful, but WITHOUT |
jlaskey@3 | 12 | * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or |
jlaskey@3 | 13 | * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License |
jlaskey@3 | 14 | * version 2 for more details (a copy is included in the LICENSE file that |
jlaskey@3 | 15 | * accompanied this code). |
jlaskey@3 | 16 | * |
jlaskey@3 | 17 | * You should have received a copy of the GNU General Public License version |
jlaskey@3 | 18 | * 2 along with this work; if not, write to the Free Software Foundation, |
jlaskey@3 | 19 | * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. |
jlaskey@3 | 20 | * |
jlaskey@3 | 21 | * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA |
jlaskey@3 | 22 | * or visit www.oracle.com if you need additional information or have any |
jlaskey@3 | 23 | * questions. |
jlaskey@3 | 24 | */ |
jlaskey@3 | 25 | |
jlaskey@3 | 26 | package jdk.nashorn.internal.objects; |
jlaskey@3 | 27 | |
jlaskey@3 | 28 | import static jdk.nashorn.internal.runtime.ECMAErrors.typeError; |
sundar@133 | 29 | import static jdk.nashorn.internal.lookup.Lookup.MH; |
jlaskey@3 | 30 | |
jlaskey@3 | 31 | import java.lang.invoke.MethodHandle; |
jlaskey@3 | 32 | import java.lang.invoke.MethodHandles; |
attila@90 | 33 | import jdk.internal.dynalink.linker.GuardedInvocation; |
attila@90 | 34 | import jdk.internal.dynalink.linker.LinkRequest; |
jlaskey@3 | 35 | import jdk.nashorn.internal.objects.annotations.Attribute; |
jlaskey@3 | 36 | import jdk.nashorn.internal.objects.annotations.Constructor; |
jlaskey@3 | 37 | import jdk.nashorn.internal.objects.annotations.Function; |
jlaskey@3 | 38 | import jdk.nashorn.internal.objects.annotations.ScriptClass; |
jlaskey@3 | 39 | import jdk.nashorn.internal.runtime.JSType; |
hannesw@380 | 40 | import jdk.nashorn.internal.runtime.PropertyMap; |
jlaskey@3 | 41 | import jdk.nashorn.internal.runtime.ScriptObject; |
jlaskey@3 | 42 | import jdk.nashorn.internal.runtime.ScriptRuntime; |
jlaskey@3 | 43 | import jdk.nashorn.internal.runtime.linker.PrimitiveLookup; |
jlaskey@3 | 44 | |
jlaskey@3 | 45 | /** |
jlaskey@3 | 46 | * ECMA 15.6 Boolean Objects. |
jlaskey@3 | 47 | */ |
jlaskey@3 | 48 | |
jlaskey@3 | 49 | @ScriptClass("Boolean") |
jlaskey@3 | 50 | public final class NativeBoolean extends ScriptObject { |
jlaskey@3 | 51 | private final boolean value; |
jlaskey@3 | 52 | |
hannesw@42 | 53 | final static MethodHandle WRAPFILTER = findWrapFilter(); |
jlaskey@3 | 54 | |
hannesw@380 | 55 | // initialized by nasgen |
hannesw@380 | 56 | private static PropertyMap $nasgenmap$; |
hannesw@380 | 57 | |
sundar@414 | 58 | static PropertyMap getInitialMap() { |
sundar@414 | 59 | return $nasgenmap$; |
jlaskey@3 | 60 | } |
jlaskey@3 | 61 | |
sundar@414 | 62 | private NativeBoolean(final boolean value, final ScriptObject proto, final PropertyMap map) { |
sundar@414 | 63 | super(proto, map); |
jlaskey@3 | 64 | this.value = value; |
jlaskey@3 | 65 | } |
jlaskey@3 | 66 | |
sundar@414 | 67 | NativeBoolean(final boolean flag, final Global global) { |
hannesw@766 | 68 | this(flag, global.getBooleanPrototype(), getInitialMap()); |
sundar@414 | 69 | } |
sundar@414 | 70 | |
sundar@414 | 71 | NativeBoolean(final boolean flag) { |
sundar@414 | 72 | this(flag, Global.instance()); |
sundar@414 | 73 | } |
sundar@414 | 74 | |
jlaskey@3 | 75 | @Override |
jlaskey@3 | 76 | public String safeToString() { |
jlaskey@3 | 77 | return "[Boolean " + toString() + "]"; |
jlaskey@3 | 78 | } |
jlaskey@3 | 79 | |
jlaskey@3 | 80 | @Override |
jlaskey@3 | 81 | public String toString() { |
jlaskey@3 | 82 | return Boolean.toString(getValue()); |
jlaskey@3 | 83 | } |
jlaskey@3 | 84 | |
jlaskey@3 | 85 | /** |
jlaskey@3 | 86 | * Get the value for this NativeBoolean |
jlaskey@3 | 87 | * @return true or false |
jlaskey@3 | 88 | */ |
jlaskey@3 | 89 | public boolean getValue() { |
jlaskey@3 | 90 | return booleanValue(); |
jlaskey@3 | 91 | } |
jlaskey@3 | 92 | |
jlaskey@3 | 93 | /** |
jlaskey@3 | 94 | * Get the value for this NativeBoolean |
jlaskey@3 | 95 | * @return true or false |
jlaskey@3 | 96 | */ |
jlaskey@3 | 97 | public boolean booleanValue() { |
jlaskey@3 | 98 | return value; |
jlaskey@3 | 99 | } |
jlaskey@3 | 100 | |
jlaskey@3 | 101 | @Override |
jlaskey@3 | 102 | public String getClassName() { |
jlaskey@3 | 103 | return "Boolean"; |
jlaskey@3 | 104 | } |
jlaskey@3 | 105 | |
jlaskey@3 | 106 | /** |
jlaskey@3 | 107 | * ECMA 15.6.4.2 Boolean.prototype.toString ( ) |
jlaskey@3 | 108 | * |
jlaskey@3 | 109 | * @param self self reference |
jlaskey@3 | 110 | * @return string representation of this boolean |
jlaskey@3 | 111 | */ |
jlaskey@3 | 112 | @Function(attributes = Attribute.NOT_ENUMERABLE) |
jlaskey@3 | 113 | public static Object toString(final Object self) { |
jlaskey@3 | 114 | return getBoolean(self).toString(); |
jlaskey@3 | 115 | } |
jlaskey@3 | 116 | |
jlaskey@3 | 117 | /** |
jlaskey@3 | 118 | * ECMA 15.6.4.3 Boolean.prototype.valueOf ( ) |
jlaskey@3 | 119 | * |
jlaskey@3 | 120 | * @param self self reference |
jlaskey@3 | 121 | * @return value of this boolean |
jlaskey@3 | 122 | */ |
jlaskey@3 | 123 | @Function(attributes = Attribute.NOT_ENUMERABLE) |
jlaskey@3 | 124 | public static Object valueOf(final Object self) { |
jlaskey@3 | 125 | return getBoolean(self); |
jlaskey@3 | 126 | } |
jlaskey@3 | 127 | |
jlaskey@3 | 128 | /** |
jlaskey@3 | 129 | * ECMA 15.6.2.1 new Boolean (value) |
jlaskey@3 | 130 | * |
jlaskey@3 | 131 | * @param newObj is the new operator used to instantiate this NativeBoolean |
jlaskey@3 | 132 | * @param self self reference |
jlaskey@3 | 133 | * @param value value of boolean |
jlaskey@3 | 134 | * @return the new NativeBoolean |
jlaskey@3 | 135 | */ |
jlaskey@3 | 136 | @Constructor(arity = 1) |
jlaskey@3 | 137 | public static Object constructor(final boolean newObj, final Object self, final Object value) { |
jlaskey@3 | 138 | final boolean flag = JSType.toBoolean(value); |
jlaskey@3 | 139 | |
jlaskey@3 | 140 | if (newObj) { |
sundar@414 | 141 | return new NativeBoolean(flag); |
jlaskey@3 | 142 | } |
jlaskey@3 | 143 | |
jlaskey@3 | 144 | return flag; |
jlaskey@3 | 145 | } |
jlaskey@3 | 146 | |
jlaskey@3 | 147 | private static Boolean getBoolean(final Object self) { |
jlaskey@3 | 148 | if (self instanceof Boolean) { |
jlaskey@3 | 149 | return ((Boolean)self); |
jlaskey@3 | 150 | } else if (self instanceof NativeBoolean) { |
jlaskey@3 | 151 | return ((NativeBoolean)self).getValue(); |
jlaskey@3 | 152 | } else if (self != null && self == Global.instance().getBooleanPrototype()) { |
jlaskey@3 | 153 | return false; |
jlaskey@3 | 154 | } else { |
lagergren@112 | 155 | throw typeError("not.a.boolean", ScriptRuntime.safeToString(self)); |
jlaskey@3 | 156 | } |
jlaskey@3 | 157 | } |
jlaskey@3 | 158 | |
jlaskey@3 | 159 | /** |
jlaskey@3 | 160 | * Lookup the appropriate method for an invoke dynamic call. |
jlaskey@3 | 161 | * |
hannesw@51 | 162 | * @param request The link request |
jlaskey@3 | 163 | * @param receiver The receiver for the call |
jlaskey@3 | 164 | * @return Link to be invoked at call site. |
jlaskey@3 | 165 | */ |
hannesw@51 | 166 | public static GuardedInvocation lookupPrimitive(final LinkRequest request, final Object receiver) { |
hannesw@51 | 167 | return PrimitiveLookup.lookupPrimitive(request, Boolean.class, new NativeBoolean((Boolean)receiver), WRAPFILTER); |
jlaskey@3 | 168 | } |
jlaskey@3 | 169 | |
jlaskey@3 | 170 | /** |
jlaskey@3 | 171 | * Wrap a native string in a NativeString object. |
jlaskey@3 | 172 | * |
jlaskey@3 | 173 | * @param receiver Native string. |
jlaskey@3 | 174 | * @return Wrapped object. |
jlaskey@3 | 175 | */ |
jlaskey@3 | 176 | @SuppressWarnings("unused") |
jlaskey@3 | 177 | private static NativeBoolean wrapFilter(final Object receiver) { |
jlaskey@3 | 178 | return new NativeBoolean((Boolean)receiver); |
jlaskey@3 | 179 | } |
jlaskey@3 | 180 | |
jlaskey@3 | 181 | private static MethodHandle findWrapFilter() { |
sundar@414 | 182 | return MH.findStatic(MethodHandles.lookup(), NativeBoolean.class, "wrapFilter", MH.type(NativeBoolean.class, Object.class)); |
jlaskey@3 | 183 | } |
jlaskey@3 | 184 | } |