8012190: Global scope should be initialized lazily

Reviewed-by: lagergren, hannesw, attila
This commit is contained in:
Athijegannathan Sundararajan 2015-03-25 14:36:22 +05:30
parent 0803c08355
commit de784d82f1
3 changed files with 521 additions and 149 deletions

View File

@ -54,8 +54,10 @@ import jdk.nashorn.api.scripting.ClassFilter;
import jdk.nashorn.api.scripting.ScriptObjectMirror; import jdk.nashorn.api.scripting.ScriptObjectMirror;
import jdk.nashorn.internal.lookup.Lookup; import jdk.nashorn.internal.lookup.Lookup;
import jdk.nashorn.internal.objects.annotations.Attribute; import jdk.nashorn.internal.objects.annotations.Attribute;
import jdk.nashorn.internal.objects.annotations.Getter;
import jdk.nashorn.internal.objects.annotations.Property; import jdk.nashorn.internal.objects.annotations.Property;
import jdk.nashorn.internal.objects.annotations.ScriptClass; import jdk.nashorn.internal.objects.annotations.ScriptClass;
import jdk.nashorn.internal.objects.annotations.Setter;
import jdk.nashorn.internal.runtime.Context; import jdk.nashorn.internal.runtime.Context;
import jdk.nashorn.internal.runtime.ECMAErrors; import jdk.nashorn.internal.runtime.ECMAErrors;
import jdk.nashorn.internal.runtime.GlobalConstants; import jdk.nashorn.internal.runtime.GlobalConstants;
@ -89,6 +91,9 @@ public final class Global extends ScriptObject implements Scope {
private final InvokeByName TO_STRING = new InvokeByName("toString", ScriptObject.class); private final InvokeByName TO_STRING = new InvokeByName("toString", ScriptObject.class);
private final InvokeByName VALUE_OF = new InvokeByName("valueOf", ScriptObject.class); private final InvokeByName VALUE_OF = new InvokeByName("valueOf", ScriptObject.class);
// placeholder value for lazily initialized global objects
private static final Object LAZY_SENTINEL = new Object();
/** /**
* Optimistic builtin names that require switchpoint invalidation * Optimistic builtin names that require switchpoint invalidation
* upon assignment. Overly conservative, but works for now, to avoid * upon assignment. Overly conservative, but works for now, to avoid
@ -214,20 +219,76 @@ public final class Global extends ScriptObject implements Scope {
public volatile Object number; public volatile Object number;
/** ECMA 15.1.4.7 Date constructor */ /** ECMA 15.1.4.7 Date constructor */
@Property(name = "Date", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "Date", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object date; public static Object getDate(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.date == LAZY_SENTINEL) {
global.date = global.getBuiltinDate();
}
return global.date;
}
@Setter(name = "Date", attributes = Attribute.NOT_ENUMERABLE)
public static void setDate(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.date = value;
}
private volatile Object date = LAZY_SENTINEL;
/** ECMA 15.1.4.8 RegExp constructor */ /** ECMA 15.1.4.8 RegExp constructor */
@Property(name = "RegExp", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "RegExp", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object regexp; public static Object getRegExp(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.regexp == LAZY_SENTINEL) {
global.regexp = global.getBuiltinRegExp();
}
return global.regexp;
}
@Setter(name = "RegExp", attributes = Attribute.NOT_ENUMERABLE)
public static void setRegExp(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.regexp = value;
}
private volatile Object regexp = LAZY_SENTINEL;
/** ECMA 15.12 - The JSON object */ /** ECMA 15.12 - The JSON object */
@Property(name = "JSON", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "JSON", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object json; public static Object getJSON(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.json == LAZY_SENTINEL) {
global.json = global.getBuiltinJSON();
}
return global.json;
}
@Setter(name = "JSON", attributes = Attribute.NOT_ENUMERABLE)
public static void setJSON(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.json = value;
}
private volatile Object json = LAZY_SENTINEL;
/** Nashorn extension: global.JSAdapter */ /** Nashorn extension: global.JSAdapter */
@Property(name = "JSAdapter", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "JSAdapter", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object jsadapter; public static Object getJSAdapter(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.jsadapter == LAZY_SENTINEL) {
global.jsadapter = global.getBuiltinJSAdapter();
}
return global.jsadapter;
}
@Setter(name = "JSAdapter", attributes = Attribute.NOT_ENUMERABLE)
public static void setJSAdapter(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.jsadapter = value;
}
private volatile Object jsadapter = LAZY_SENTINEL;
/** ECMA 15.8 - The Math object */ /** ECMA 15.8 - The Math object */
@Property(name = "Math", attributes = Attribute.NOT_ENUMERABLE) @Property(name = "Math", attributes = Attribute.NOT_ENUMERABLE)
@ -238,12 +299,40 @@ public final class Global extends ScriptObject implements Scope {
public volatile Object error; public volatile Object error;
/** EvalError object */ /** EvalError object */
@Property(name = "EvalError", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "EvalError", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object evalError; public static Object getEvalError(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.evalError == LAZY_SENTINEL) {
global.evalError = global.getBuiltinEvalError();
}
return global.evalError;
}
@Setter(name = "EvalError", attributes = Attribute.NOT_ENUMERABLE)
public static void setEvalError(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.evalError = value;
}
private volatile Object evalError = LAZY_SENTINEL;
/** RangeError object */ /** RangeError object */
@Property(name = "RangeError", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "RangeError", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object rangeError; public static Object getRangeError(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.rangeError == LAZY_SENTINEL) {
global.rangeError = global.getBuiltinRangeError();
}
return global.rangeError;
}
@Setter(name = "RangeError", attributes = Attribute.NOT_ENUMERABLE)
public static void setRangeError(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.rangeError = value;
}
private volatile Object rangeError = LAZY_SENTINEL;
/** ReferenceError object */ /** ReferenceError object */
@Property(name = "ReferenceError", attributes = Attribute.NOT_ENUMERABLE) @Property(name = "ReferenceError", attributes = Attribute.NOT_ENUMERABLE)
@ -258,52 +347,220 @@ public final class Global extends ScriptObject implements Scope {
public volatile Object typeError; public volatile Object typeError;
/** URIError object */ /** URIError object */
@Property(name = "URIError", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "URIError", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object uriError; public static Object getURIError(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.uriError == LAZY_SENTINEL) {
global.uriError = global.getBuiltinURIError();
}
return global.uriError;
}
@Setter(name = "URIError", attributes = Attribute.NOT_ENUMERABLE)
public static void setURIError(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.uriError = value;
}
private volatile Object uriError = LAZY_SENTINEL;
/** ArrayBuffer object */ /** ArrayBuffer object */
@Property(name = "ArrayBuffer", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "ArrayBuffer", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object arrayBuffer; public static Object getArrayBuffer(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.arrayBuffer == LAZY_SENTINEL) {
global.arrayBuffer = global.getBuiltinArrayBuffer();
}
return global.arrayBuffer;
}
@Setter(name = "ArrayBuffer", attributes = Attribute.NOT_ENUMERABLE)
public static void setArrayBuffer(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.arrayBuffer = value;
}
private volatile Object arrayBuffer;
/** DataView object */ /** DataView object */
@Property(name = "DataView", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "DataView", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object dataView; public static Object getDataView(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.dataView == LAZY_SENTINEL) {
global.dataView = global.getBuiltinDataView();
}
return global.dataView;
}
@Setter(name = "DataView", attributes = Attribute.NOT_ENUMERABLE)
public static void setDataView(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.dataView = value;
}
private volatile Object dataView;
/** TypedArray (int8) */ /** TypedArray (int8) */
@Property(name = "Int8Array", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "Int8Array", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object int8Array; public static Object getInt8Array(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.int8Array == LAZY_SENTINEL) {
global.int8Array = global.getBuiltinInt8Array();
}
return global.int8Array;
}
@Setter(name = "Int8Array", attributes = Attribute.NOT_ENUMERABLE)
public static void setInt8Array(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.int8Array = value;
}
private volatile Object int8Array;
/** TypedArray (uint8) */ /** TypedArray (uint8) */
@Property(name = "Uint8Array", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "Uint8Array", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object uint8Array; public static Object getUint8Array(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.uint8Array == LAZY_SENTINEL) {
global.uint8Array = global.getBuiltinUint8Array();
}
return global.uint8Array;
}
@Setter(name = "Uint8Array", attributes = Attribute.NOT_ENUMERABLE)
public static void setUint8Array(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.uint8Array = value;
}
private volatile Object uint8Array;
/** TypedArray (uint8) - Clamped */ /** TypedArray (uint8) - Clamped */
@Property(name = "Uint8ClampedArray", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "Uint8ClampedArray", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object uint8ClampedArray; public static Object getUint8ClampedArray(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.uint8ClampedArray == LAZY_SENTINEL) {
global.uint8ClampedArray = global.getBuiltinUint8ClampedArray();
}
return global.uint8ClampedArray;
}
@Setter(name = "Uint8ClampedArray", attributes = Attribute.NOT_ENUMERABLE)
public static void setUint8ClampedArray(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.uint8ClampedArray = value;
}
private volatile Object uint8ClampedArray;
/** TypedArray (int16) */ /** TypedArray (int16) */
@Property(name = "Int16Array", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "Int16Array", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object int16Array; public static Object getInt16Array(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.int16Array == LAZY_SENTINEL) {
global.int16Array = global.getBuiltinInt16Array();
}
return global.int16Array;
}
@Setter(name = "Int16Array", attributes = Attribute.NOT_ENUMERABLE)
public static void setInt16Array(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.int16Array = value;
}
private volatile Object int16Array;
/** TypedArray (uint16) */ /** TypedArray (uint16) */
@Property(name = "Uint16Array", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "Uint16Array", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object uint16Array; public static Object getUint16Array(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.uint16Array == LAZY_SENTINEL) {
global.uint16Array = global.getBuiltinUint16Array();
}
return global.uint16Array;
}
@Setter(name = "Uint16Array", attributes = Attribute.NOT_ENUMERABLE)
public static void setUint16Array(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.uint16Array = value;
}
private volatile Object uint16Array;
/** TypedArray (int32) */ /** TypedArray (int32) */
@Property(name = "Int32Array", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "Int32Array", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object int32Array; public static Object getInt32Array(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.int32Array == LAZY_SENTINEL) {
global.int32Array = global.getBuiltinInt32Array();
}
return global.int32Array;
}
@Setter(name = "Int32Array", attributes = Attribute.NOT_ENUMERABLE)
public static void setInt32Array(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.int32Array = value;
}
private volatile Object int32Array;
/** TypedArray (uint32) */ /** TypedArray (uint32) */
@Property(name = "Uint32Array", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "Uint32Array", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object uint32Array; public static Object getUint32Array(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.uint32Array == LAZY_SENTINEL) {
global.uint32Array = global.getBuiltinUint32Array();
}
return global.uint32Array;
}
@Setter(name = "Uint32Array", attributes = Attribute.NOT_ENUMERABLE)
public static void setUint32Array(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.uint32Array = value;
}
private volatile Object uint32Array;
/** TypedArray (float32) */ /** TypedArray (float32) */
@Property(name = "Float32Array", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "Float32Array", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object float32Array; public static Object getFloat32Array(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.float32Array == LAZY_SENTINEL) {
global.float32Array = global.getBuiltinFloat32Array();
}
return global.float32Array;
}
@Setter(name = "Float32Array", attributes = Attribute.NOT_ENUMERABLE)
public static void setFloat32Array(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.float32Array = value;
}
private volatile Object float32Array;
/** TypedArray (float64) */ /** TypedArray (float64) */
@Property(name = "Float64Array", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "Float64Array", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object float64Array; public static Object getFloat64Array(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.float64Array == LAZY_SENTINEL) {
global.float64Array = global.getBuiltinFloat64Array();
}
return global.float64Array;
}
@Setter(name = "Float64Array", attributes = Attribute.NOT_ENUMERABLE)
public static void setFloat64Array(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.float64Array = value;
}
private volatile Object float64Array;
/** Nashorn extension: Java access - global.Packages */ /** Nashorn extension: Java access - global.Packages */
@Property(name = "Packages", attributes = Attribute.NOT_ENUMERABLE) @Property(name = "Packages", attributes = Attribute.NOT_ENUMERABLE)
@ -334,12 +591,40 @@ public final class Global extends ScriptObject implements Scope {
public volatile Object org; public volatile Object org;
/** Nashorn extension: Java access - global.javaImporter */ /** Nashorn extension: Java access - global.javaImporter */
@Property(name = "JavaImporter", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "JavaImporter", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object javaImporter; public static Object getJavaImporter(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.javaImporter == LAZY_SENTINEL) {
global.javaImporter = global.getBuiltinJavaImporter();
}
return global.javaImporter;
}
@Setter(name = "JavaImporter", attributes = Attribute.NOT_ENUMERABLE)
public static void setJavaImporter(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.javaImporter = value;
}
private volatile Object javaImporter;
/** Nashorn extension: global.Java Object constructor. */ /** Nashorn extension: global.Java Object constructor. */
@Property(name = "Java", attributes = Attribute.NOT_ENUMERABLE) @Getter(name = "Java", attributes = Attribute.NOT_ENUMERABLE)
public volatile Object javaApi; public static Object getJavaApi(final Object self) {
final Global global = Global.instanceFrom(self);
if (global.javaApi == LAZY_SENTINEL) {
global.javaApi = global.getBuiltinJavaApi();
}
return global.javaApi;
}
@Setter(name = "Java", attributes = Attribute.NOT_ENUMERABLE)
public static void setJavaApi(final Object self, final Object value) {
final Global global = Global.instanceFrom(self);
global.javaApi = value;
}
private volatile Object javaApi;
/** Nashorn extension: current script's file name */ /** Nashorn extension: current script's file name */
@Property(name = "__FILE__", attributes = Attribute.NON_ENUMERABLE_CONSTANT) @Property(name = "__FILE__", attributes = Attribute.NON_ENUMERABLE_CONSTANT)
@ -353,11 +638,19 @@ public final class Global extends ScriptObject implements Scope {
@Property(name = "__LINE__", attributes = Attribute.NON_ENUMERABLE_CONSTANT) @Property(name = "__LINE__", attributes = Attribute.NON_ENUMERABLE_CONSTANT)
public final Object __LINE__ = LOCATION_PROPERTY_PLACEHOLDER; public final Object __LINE__ = LOCATION_PROPERTY_PLACEHOLDER;
private volatile NativeDate DEFAULT_DATE;
/** Used as Date.prototype's default value */ /** Used as Date.prototype's default value */
public NativeDate DEFAULT_DATE; NativeDate getDefaultDate() {
return DEFAULT_DATE;
}
private volatile NativeRegExp DEFAULT_REGEXP;
/** Used as RegExp.prototype's default value */ /** Used as RegExp.prototype's default value */
public NativeRegExp DEFAULT_REGEXP; NativeRegExp getDefaultRegExp() {
return DEFAULT_REGEXP;
}
/* /*
* Built-in constructor objects: Even if user changes dynamic values of * Built-in constructor objects: Even if user changes dynamic values of
@ -1034,7 +1327,7 @@ public final class Global extends ScriptObject implements Scope {
/** /**
* Get the builtin Object prototype. * Get the builtin Object prototype.
* @return the object prototype. * @return the object prototype.
*/ */
public ScriptObject getObjectPrototype() { public ScriptObject getObjectPrototype() {
return ScriptFunction.getPrototype(builtinObject); return ScriptFunction.getPrototype(builtinObject);
@ -1057,11 +1350,11 @@ public final class Global extends ScriptObject implements Scope {
} }
ScriptObject getDatePrototype() { ScriptObject getDatePrototype() {
return ScriptFunction.getPrototype(builtinDate); return ScriptFunction.getPrototype(getBuiltinDate());
} }
ScriptObject getRegExpPrototype() { ScriptObject getRegExpPrototype() {
return ScriptFunction.getPrototype(builtinRegExp); return ScriptFunction.getPrototype(getBuiltinRegExp());
} }
ScriptObject getStringPrototype() { ScriptObject getStringPrototype() {
@ -1073,11 +1366,11 @@ public final class Global extends ScriptObject implements Scope {
} }
ScriptObject getEvalErrorPrototype() { ScriptObject getEvalErrorPrototype() {
return ScriptFunction.getPrototype(builtinEvalError); return ScriptFunction.getPrototype(getBuiltinEvalError());
} }
ScriptObject getRangeErrorPrototype() { ScriptObject getRangeErrorPrototype() {
return ScriptFunction.getPrototype(builtinRangeError); return ScriptFunction.getPrototype(getBuiltinRangeError());
} }
ScriptObject getReferenceErrorPrototype() { ScriptObject getReferenceErrorPrototype() {
@ -1093,59 +1386,136 @@ public final class Global extends ScriptObject implements Scope {
} }
ScriptObject getURIErrorPrototype() { ScriptObject getURIErrorPrototype() {
return ScriptFunction.getPrototype(builtinURIError); return ScriptFunction.getPrototype(getBuiltinURIError());
} }
ScriptObject getJavaImporterPrototype() { ScriptObject getJavaImporterPrototype() {
return ScriptFunction.getPrototype(builtinJavaImporter); return ScriptFunction.getPrototype(getBuiltinJavaImporter());
} }
ScriptObject getJSAdapterPrototype() { ScriptObject getJSAdapterPrototype() {
return ScriptFunction.getPrototype(builtinJSAdapter); return ScriptFunction.getPrototype(getBuiltinJSAdapter());
}
private synchronized ScriptFunction getBuiltinArrayBuffer() {
if (this.builtinArrayBuffer == null) {
this.builtinArrayBuffer = initConstructorAndSwitchPoint("ArrayBuffer", ScriptFunction.class);
}
return this.builtinArrayBuffer;
} }
ScriptObject getArrayBufferPrototype() { ScriptObject getArrayBufferPrototype() {
return ScriptFunction.getPrototype(builtinArrayBuffer); return ScriptFunction.getPrototype(getBuiltinArrayBuffer());
}
private synchronized ScriptFunction getBuiltinDataView() {
if (this.builtinDataView == null) {
this.builtinDataView = initConstructorAndSwitchPoint("DataView", ScriptFunction.class);
}
return this.builtinDataView;
} }
ScriptObject getDataViewPrototype() { ScriptObject getDataViewPrototype() {
return ScriptFunction.getPrototype(builtinDataView); return ScriptFunction.getPrototype(getBuiltinDataView());
}
private synchronized ScriptFunction getBuiltinInt8Array() {
if (this.builtinInt8Array == null) {
this.builtinInt8Array = initConstructorAndSwitchPoint("Int8Array", ScriptFunction.class);
}
return this.builtinInt8Array;
} }
ScriptObject getInt8ArrayPrototype() { ScriptObject getInt8ArrayPrototype() {
return ScriptFunction.getPrototype(builtinInt8Array); return ScriptFunction.getPrototype(getBuiltinInt8Array());
}
private synchronized ScriptFunction getBuiltinUint8Array() {
if (this.builtinUint8Array == null) {
this.builtinUint8Array = initConstructorAndSwitchPoint("Uint8Array", ScriptFunction.class);
}
return this.builtinUint8Array;
} }
ScriptObject getUint8ArrayPrototype() { ScriptObject getUint8ArrayPrototype() {
return ScriptFunction.getPrototype(builtinUint8Array); return ScriptFunction.getPrototype(getBuiltinUint8Array());
}
private synchronized ScriptFunction getBuiltinUint8ClampedArray() {
if (this.builtinUint8ClampedArray == null) {
this.builtinUint8ClampedArray = initConstructorAndSwitchPoint("Uint8ClampedArray", ScriptFunction.class);
}
return this.builtinUint8ClampedArray;
} }
ScriptObject getUint8ClampedArrayPrototype() { ScriptObject getUint8ClampedArrayPrototype() {
return ScriptFunction.getPrototype(builtinUint8ClampedArray); return ScriptFunction.getPrototype(getBuiltinUint8ClampedArray());
}
private synchronized ScriptFunction getBuiltinInt16Array() {
if (this.builtinInt16Array == null) {
this.builtinInt16Array = initConstructorAndSwitchPoint("Int16Array", ScriptFunction.class);
}
return this.builtinInt16Array;
} }
ScriptObject getInt16ArrayPrototype() { ScriptObject getInt16ArrayPrototype() {
return ScriptFunction.getPrototype(builtinInt16Array); return ScriptFunction.getPrototype(getBuiltinInt16Array());
}
private synchronized ScriptFunction getBuiltinUint16Array() {
if (this.builtinUint16Array == null) {
this.builtinUint16Array = initConstructorAndSwitchPoint("Uint16Array", ScriptFunction.class);
}
return this.builtinUint16Array;
} }
ScriptObject getUint16ArrayPrototype() { ScriptObject getUint16ArrayPrototype() {
return ScriptFunction.getPrototype(builtinUint16Array); return ScriptFunction.getPrototype(getBuiltinUint16Array());
}
private synchronized ScriptFunction getBuiltinInt32Array() {
if (this.builtinInt32Array == null) {
this.builtinInt32Array = initConstructorAndSwitchPoint("Int32Array", ScriptFunction.class);
}
return this.builtinInt32Array;
} }
ScriptObject getInt32ArrayPrototype() { ScriptObject getInt32ArrayPrototype() {
return ScriptFunction.getPrototype(builtinInt32Array); return ScriptFunction.getPrototype(getBuiltinInt32Array());
}
private synchronized ScriptFunction getBuiltinUint32Array() {
if (this.builtinUint32Array == null) {
this.builtinUint32Array = initConstructorAndSwitchPoint("Uint32Array", ScriptFunction.class);
}
return this.builtinUint32Array;
} }
ScriptObject getUint32ArrayPrototype() { ScriptObject getUint32ArrayPrototype() {
return ScriptFunction.getPrototype(builtinUint32Array); return ScriptFunction.getPrototype(getBuiltinUint32Array());
}
private synchronized ScriptFunction getBuiltinFloat32Array() {
if (this.builtinFloat32Array == null) {
this.builtinFloat32Array = initConstructorAndSwitchPoint("Float32Array", ScriptFunction.class);
}
return this.builtinFloat32Array;
} }
ScriptObject getFloat32ArrayPrototype() { ScriptObject getFloat32ArrayPrototype() {
return ScriptFunction.getPrototype(builtinFloat32Array); return ScriptFunction.getPrototype(getBuiltinFloat32Array());
}
private synchronized ScriptFunction getBuiltinFloat64Array() {
if (this.builtinFloat64Array == null) {
this.builtinFloat64Array = initConstructorAndSwitchPoint("Float64Array", ScriptFunction.class);
}
return this.builtinFloat64Array;
} }
ScriptObject getFloat64ArrayPrototype() { ScriptObject getFloat64ArrayPrototype() {
return ScriptFunction.getPrototype(builtinFloat64Array); return ScriptFunction.getPrototype(getBuiltinFloat64Array());
} }
private ScriptFunction getBuiltinArray() { private ScriptFunction getBuiltinArray() {
@ -1180,8 +1550,14 @@ public final class Global extends ScriptObject implements Scope {
return instance._boolean == instance.getBuiltinBoolean(); return instance._boolean == instance.getBuiltinBoolean();
} }
private ScriptFunction getBuiltinDate() { private synchronized ScriptFunction getBuiltinDate() {
return builtinDate; if (this.builtinDate == null) {
this.builtinDate = initConstructorAndSwitchPoint("Date", ScriptFunction.class);
final ScriptObject dateProto = ScriptFunction.getPrototype(builtinDate);
// initialize default date
this.DEFAULT_DATE = new NativeDate(NaN, dateProto);
}
return this.builtinDate;
} }
/** /**
@ -1191,7 +1567,7 @@ public final class Global extends ScriptObject implements Scope {
*/ */
public static boolean isBuiltinDate() { public static boolean isBuiltinDate() {
final Global instance = Global.instance(); final Global instance = Global.instance();
return instance.date == instance.getBuiltinDate(); return instance.date == LAZY_SENTINEL || instance.date == instance.getBuiltinDate();
} }
private ScriptFunction getBuiltinError() { private ScriptFunction getBuiltinError() {
@ -1208,8 +1584,11 @@ public final class Global extends ScriptObject implements Scope {
return instance.error == instance.getBuiltinError(); return instance.error == instance.getBuiltinError();
} }
private ScriptFunction getBuiltinEvalError() { private synchronized ScriptFunction getBuiltinEvalError() {
return builtinEvalError; if (this.builtinEvalError == null) {
this.builtinEvalError = initErrorSubtype("EvalError", getErrorPrototype());
}
return this.builtinEvalError;
} }
/** /**
@ -1219,7 +1598,7 @@ public final class Global extends ScriptObject implements Scope {
*/ */
public static boolean isBuiltinEvalError() { public static boolean isBuiltinEvalError() {
final Global instance = Global.instance(); final Global instance = Global.instance();
return instance.evalError == instance.getBuiltinEvalError(); return instance.evalError == LAZY_SENTINEL || instance.evalError == instance.getBuiltinEvalError();
} }
private ScriptFunction getBuiltinFunction() { private ScriptFunction getBuiltinFunction() {
@ -1270,7 +1649,10 @@ public final class Global extends ScriptObject implements Scope {
return isBuiltinFunctionProperty("call"); return isBuiltinFunctionProperty("call");
} }
private ScriptFunction getBuiltinJSAdapter() { private synchronized ScriptFunction getBuiltinJSAdapter() {
if (this.builtinJSAdapter == null) {
this.builtinJSAdapter = initConstructorAndSwitchPoint("JSAdapter", ScriptFunction.class);
}
return builtinJSAdapter; return builtinJSAdapter;
} }
@ -1281,11 +1663,14 @@ public final class Global extends ScriptObject implements Scope {
*/ */
public static boolean isBuiltinJSAdapter() { public static boolean isBuiltinJSAdapter() {
final Global instance = Global.instance(); final Global instance = Global.instance();
return instance.jsadapter == instance.getBuiltinJSAdapter(); return instance.jsadapter == LAZY_SENTINEL || instance.jsadapter == instance.getBuiltinJSAdapter();
} }
private ScriptObject getBuiltinJSON() { private synchronized ScriptObject getBuiltinJSON() {
return builtinJSON; if (this.builtinJSON == null) {
this.builtinJSON = initConstructorAndSwitchPoint("JSON", ScriptObject.class);
}
return this.builtinJSON;
} }
/** /**
@ -1295,7 +1680,7 @@ public final class Global extends ScriptObject implements Scope {
*/ */
public static boolean isBuiltinJSON() { public static boolean isBuiltinJSON() {
final Global instance = Global.instance(); final Global instance = Global.instance();
return instance.json == instance.getBuiltinJSON(); return instance.json == LAZY_SENTINEL || instance.json == instance.getBuiltinJSON();
} }
private ScriptObject getBuiltinJava() { private ScriptObject getBuiltinJava() {
@ -1326,8 +1711,18 @@ public final class Global extends ScriptObject implements Scope {
return instance.javax == instance.getBuiltinJavax(); return instance.javax == instance.getBuiltinJavax();
} }
private ScriptObject getBuiltinJavaImporter() { private synchronized ScriptFunction getBuiltinJavaImporter() {
return builtinJavaImporter; if (this.builtinJavaImporter == null) {
this.builtinJavaImporter = initConstructor("JavaImporter", ScriptFunction.class);
}
return this.builtinJavaImporter;
}
private synchronized ScriptObject getBuiltinJavaApi() {
if (this.builtinJavaApi == null) {
this.builtinJavaApi = initConstructor("Java", ScriptObject.class);
}
return this.builtinJavaApi;
} }
/** /**
@ -1337,11 +1732,7 @@ public final class Global extends ScriptObject implements Scope {
*/ */
public static boolean isBuiltinJavaImporter() { public static boolean isBuiltinJavaImporter() {
final Global instance = Global.instance(); final Global instance = Global.instance();
return instance.javaImporter == instance.getBuiltinJavaImporter(); return instance.javaImporter == LAZY_SENTINEL || instance.javaImporter == instance.getBuiltinJavaImporter();
}
private ScriptObject getBuiltinMath() {
return builtinMath;
} }
/** /**
@ -1351,7 +1742,7 @@ public final class Global extends ScriptObject implements Scope {
*/ */
public static boolean isBuiltinMath() { public static boolean isBuiltinMath() {
final Global instance = Global.instance(); final Global instance = Global.instance();
return instance.math == instance.getBuiltinMath(); return instance.math == instance.builtinMath;
} }
private ScriptFunction getBuiltinNumber() { private ScriptFunction getBuiltinNumber() {
@ -1396,7 +1787,10 @@ public final class Global extends ScriptObject implements Scope {
return instance.packages == instance.getBuiltinPackages(); return instance.packages == instance.getBuiltinPackages();
} }
private ScriptFunction getBuiltinRangeError() { private synchronized ScriptFunction getBuiltinRangeError() {
if (this.builtinRangeError == null) {
this.builtinRangeError = initErrorSubtype("RangeError", getErrorPrototype());
}
return builtinRangeError; return builtinRangeError;
} }
@ -1407,10 +1801,10 @@ public final class Global extends ScriptObject implements Scope {
*/ */
public static boolean isBuiltinRangeError() { public static boolean isBuiltinRangeError() {
final Global instance = Global.instance(); final Global instance = Global.instance();
return instance.rangeError == instance.getBuiltinRangeError(); return instance.rangeError == LAZY_SENTINEL || instance.rangeError == instance.getBuiltinRangeError();
} }
private ScriptFunction getBuiltinReferenceError() { private synchronized ScriptFunction getBuiltinReferenceError() {
return builtinReferenceError; return builtinReferenceError;
} }
@ -1424,7 +1818,16 @@ public final class Global extends ScriptObject implements Scope {
return instance.referenceError == instance.getBuiltinReferenceError(); return instance.referenceError == instance.getBuiltinReferenceError();
} }
private ScriptFunction getBuiltinRegExp() { private synchronized ScriptFunction getBuiltinRegExp() {
if (this.builtinRegExp == null) {
this.builtinRegExp = initConstructorAndSwitchPoint("RegExp", ScriptFunction.class);
final ScriptObject regExpProto = ScriptFunction.getPrototype(builtinRegExp);
// initialize default regexp object
this.DEFAULT_REGEXP = new NativeRegExp("(?:)", "", this, regExpProto);
// RegExp.prototype should behave like a RegExp object. So copy the
// properties.
regExpProto.addBoundProperties(DEFAULT_REGEXP);
}
return builtinRegExp; return builtinRegExp;
} }
@ -1435,7 +1838,7 @@ public final class Global extends ScriptObject implements Scope {
*/ */
public static boolean isBuiltinRegExp() { public static boolean isBuiltinRegExp() {
final Global instance = Global.instance(); final Global instance = Global.instance();
return instance.regexp == instance.getBuiltinRegExp(); return instance.regexp == LAZY_SENTINEL || instance.regexp == instance.getBuiltinRegExp();
} }
private ScriptFunction getBuiltinString() { private ScriptFunction getBuiltinString() {
@ -1480,8 +1883,11 @@ public final class Global extends ScriptObject implements Scope {
return instance.typeError == instance.getBuiltinTypeError(); return instance.typeError == instance.getBuiltinTypeError();
} }
private ScriptFunction getBuiltinURIError() { private synchronized ScriptFunction getBuiltinURIError() {
return builtinURIError; if (this.builtinURIError == null) {
this.builtinURIError = initErrorSubtype("URIError", getErrorPrototype());
}
return this.builtinURIError;
} }
/** /**
@ -1491,7 +1897,7 @@ public final class Global extends ScriptObject implements Scope {
*/ */
public static boolean isBuiltinURIError() { public static boolean isBuiltinURIError() {
final Global instance = Global.instance(); final Global instance = Global.instance();
return instance.uriError == instance.getBuiltinURIError(); return instance.uriError == LAZY_SENTINEL || instance.uriError == instance.getBuiltinURIError();
} }
@Override @Override
@ -1903,13 +2309,9 @@ public final class Global extends ScriptObject implements Scope {
// built-in constructors // built-in constructors
this.builtinArray = initConstructorAndSwitchPoint("Array", ScriptFunction.class); this.builtinArray = initConstructorAndSwitchPoint("Array", ScriptFunction.class);
this.builtinBoolean = initConstructorAndSwitchPoint("Boolean", ScriptFunction.class); this.builtinBoolean = initConstructorAndSwitchPoint("Boolean", ScriptFunction.class);
this.builtinDate = initConstructorAndSwitchPoint("Date", ScriptFunction.class);
this.builtinJSON = initConstructorAndSwitchPoint("JSON", ScriptObject.class);
this.builtinJSAdapter = initConstructorAndSwitchPoint("JSAdapter", ScriptFunction.class);
this.builtinMath = initConstructorAndSwitchPoint("Math", ScriptObject.class);
this.builtinNumber = initConstructorAndSwitchPoint("Number", ScriptFunction.class); this.builtinNumber = initConstructorAndSwitchPoint("Number", ScriptFunction.class);
this.builtinRegExp = initConstructorAndSwitchPoint("RegExp", ScriptFunction.class);
this.builtinString = initConstructorAndSwitchPoint("String", ScriptFunction.class); this.builtinString = initConstructorAndSwitchPoint("String", ScriptFunction.class);
this.builtinMath = initConstructorAndSwitchPoint("Math", ScriptObject.class);
// initialize String.prototype.length to 0 // initialize String.prototype.length to 0
// add String.prototype.length // add String.prototype.length
@ -1920,26 +2322,28 @@ public final class Global extends ScriptObject implements Scope {
final ScriptObject arrayPrototype = getArrayPrototype(); final ScriptObject arrayPrototype = getArrayPrototype();
arrayPrototype.setIsArray(); arrayPrototype.setIsArray();
this.DEFAULT_DATE = new NativeDate(Double.NaN, this);
// initialize default regexp object
this.DEFAULT_REGEXP = new NativeRegExp("(?:)", this);
// RegExp.prototype should behave like a RegExp object. So copy the
// properties.
final ScriptObject regExpProto = getRegExpPrototype();
regExpProto.addBoundProperties(DEFAULT_REGEXP);
// Error stuff // Error stuff
initErrorObjects(); initErrorObjects();
// java access // java access
if (! env._no_java) { if (! env._no_java) {
this.javaApi = LAZY_SENTINEL;
this.javaImporter = LAZY_SENTINEL;
initJavaAccess(); initJavaAccess();
} }
if (! env._no_typed_arrays) { if (! env._no_typed_arrays) {
initTypedArray(); this.arrayBuffer = LAZY_SENTINEL;
this.dataView = LAZY_SENTINEL;
this.int8Array = LAZY_SENTINEL;
this.uint8Array = LAZY_SENTINEL;
this.uint8ClampedArray = LAZY_SENTINEL;
this.int16Array = LAZY_SENTINEL;
this.uint16Array = LAZY_SENTINEL;
this.int32Array = LAZY_SENTINEL;
this.uint32Array = LAZY_SENTINEL;
this.float32Array = LAZY_SENTINEL;
this.float64Array = LAZY_SENTINEL;
} }
if (env._scripting) { if (env._scripting) {
@ -2013,12 +2417,9 @@ public final class Global extends ScriptObject implements Scope {
tagBuiltinProperties("Error", builtinError); tagBuiltinProperties("Error", builtinError);
this.builtinEvalError = initErrorSubtype("EvalError", errorProto);
this.builtinRangeError = initErrorSubtype("RangeError", errorProto);
this.builtinReferenceError = initErrorSubtype("ReferenceError", errorProto); this.builtinReferenceError = initErrorSubtype("ReferenceError", errorProto);
this.builtinSyntaxError = initErrorSubtype("SyntaxError", errorProto); this.builtinSyntaxError = initErrorSubtype("SyntaxError", errorProto);
this.builtinTypeError = initErrorSubtype("TypeError", errorProto); this.builtinTypeError = initErrorSubtype("TypeError", errorProto);
this.builtinURIError = initErrorSubtype("URIError", errorProto);
} }
private ScriptFunction initErrorSubtype(final String name, final ScriptObject errorProto) { private ScriptFunction initErrorSubtype(final String name, final ScriptObject errorProto) {
@ -2040,8 +2441,6 @@ public final class Global extends ScriptObject implements Scope {
this.builtinJavafx = new NativeJavaPackage("javafx", objectProto); this.builtinJavafx = new NativeJavaPackage("javafx", objectProto);
this.builtinJavax = new NativeJavaPackage("javax", objectProto); this.builtinJavax = new NativeJavaPackage("javax", objectProto);
this.builtinOrg = new NativeJavaPackage("org", objectProto); this.builtinOrg = new NativeJavaPackage("org", objectProto);
this.builtinJavaImporter = initConstructor("JavaImporter", ScriptFunction.class);
this.builtinJavaApi = initConstructor("Java", ScriptObject.class);
} }
private void initScripting(final ScriptEnvironment scriptEnv) { private void initScripting(final ScriptEnvironment scriptEnv) {
@ -2093,60 +2492,25 @@ public final class Global extends ScriptObject implements Scope {
} }
} }
private void initTypedArray() {
this.builtinArrayBuffer = initConstructorAndSwitchPoint("ArrayBuffer", ScriptFunction.class);
this.builtinDataView = initConstructorAndSwitchPoint("DataView", ScriptFunction.class);
this.builtinInt8Array = initConstructorAndSwitchPoint("Int8Array", ScriptFunction.class);
this.builtinUint8Array = initConstructorAndSwitchPoint("Uint8Array", ScriptFunction.class);
this.builtinUint8ClampedArray = initConstructorAndSwitchPoint("Uint8ClampedArray", ScriptFunction.class);
this.builtinInt16Array = initConstructorAndSwitchPoint("Int16Array", ScriptFunction.class);
this.builtinUint16Array = initConstructorAndSwitchPoint("Uint16Array", ScriptFunction.class);
this.builtinInt32Array = initConstructorAndSwitchPoint("Int32Array", ScriptFunction.class);
this.builtinUint32Array = initConstructorAndSwitchPoint("Uint32Array", ScriptFunction.class);
this.builtinFloat32Array = initConstructorAndSwitchPoint("Float32Array", ScriptFunction.class);
this.builtinFloat64Array = initConstructorAndSwitchPoint("Float64Array", ScriptFunction.class);
}
private void copyBuiltins() { private void copyBuiltins() {
this.array = this.builtinArray; this.array = this.builtinArray;
this._boolean = this.builtinBoolean; this._boolean = this.builtinBoolean;
this.date = this.builtinDate;
this.error = this.builtinError; this.error = this.builtinError;
this.evalError = this.builtinEvalError;
this.function = this.builtinFunction; this.function = this.builtinFunction;
this.jsadapter = this.builtinJSAdapter;
this.json = this.builtinJSON;
this.com = this.builtinCom; this.com = this.builtinCom;
this.edu = this.builtinEdu; this.edu = this.builtinEdu;
this.java = this.builtinJava; this.java = this.builtinJava;
this.javafx = this.builtinJavafx; this.javafx = this.builtinJavafx;
this.javax = this.builtinJavax; this.javax = this.builtinJavax;
this.org = this.builtinOrg; this.org = this.builtinOrg;
this.javaImporter = this.builtinJavaImporter;
this.javaApi = this.builtinJavaApi;
this.math = this.builtinMath; this.math = this.builtinMath;
this.number = this.builtinNumber; this.number = this.builtinNumber;
this.object = this.builtinObject; this.object = this.builtinObject;
this.packages = this.builtinPackages; this.packages = this.builtinPackages;
this.rangeError = this.builtinRangeError;
this.referenceError = this.builtinReferenceError; this.referenceError = this.builtinReferenceError;
this.regexp = this.builtinRegExp;
this.string = this.builtinString; this.string = this.builtinString;
this.syntaxError = this.builtinSyntaxError; this.syntaxError = this.builtinSyntaxError;
this.typeError = this.builtinTypeError; this.typeError = this.builtinTypeError;
this.uriError = this.builtinURIError;
this.arrayBuffer = this.builtinArrayBuffer;
this.dataView = this.builtinDataView;
this.int8Array = this.builtinInt8Array;
this.uint8Array = this.builtinUint8Array;
this.uint8ClampedArray = this.builtinUint8ClampedArray;
this.int16Array = this.builtinInt16Array;
this.uint16Array = this.builtinUint16Array;
this.int32Array = this.builtinInt32Array;
this.uint32Array = this.builtinUint32Array;
this.float32Array = this.builtinFloat32Array;
this.float64Array = this.builtinFloat64Array;
} }
private void initDebug() { private void initDebug() {

View File

@ -121,6 +121,10 @@ public final class NativeDate extends ScriptObject {
this.timezone = env._timezone; this.timezone = env._timezone;
} }
NativeDate(final double time, final ScriptObject proto) {
this(time, proto, $nasgenmap$);
}
NativeDate(final double time, final Global global) { NativeDate(final double time, final Global global) {
this(time, global.getDatePrototype(), $nasgenmap$); this(time, global.getDatePrototype(), $nasgenmap$);
} }
@ -1276,7 +1280,7 @@ public final class NativeDate extends ScriptObject {
if (self instanceof NativeDate) { if (self instanceof NativeDate) {
return (NativeDate)self; return (NativeDate)self;
} else if (self != null && self == Global.instance().getDatePrototype()) { } else if (self != null && self == Global.instance().getDatePrototype()) {
return Global.instance().DEFAULT_DATE; return Global.instance().getDefaultDate();
} else { } else {
throw typeError("not.a.date", ScriptRuntime.safeToString(self)); throw typeError("not.a.date", ScriptRuntime.safeToString(self));
} }

View File

@ -78,8 +78,8 @@ public final class NativeRegExp extends ScriptObject {
this.globalObject = global; this.globalObject = global;
} }
NativeRegExp(final String input, final String flagString, final Global global) { NativeRegExp(final String input, final String flagString, final Global global, final ScriptObject proto) {
this(global); super(proto, $nasgenmap$);
try { try {
this.regexp = RegExpFactory.create(input, flagString); this.regexp = RegExpFactory.create(input, flagString);
} catch (final ParserException e) { } catch (final ParserException e) {
@ -87,10 +87,14 @@ public final class NativeRegExp extends ScriptObject {
e.throwAsEcmaException(); e.throwAsEcmaException();
throw new AssertionError(); //guard against null warnings below throw new AssertionError(); //guard against null warnings below
} }
this.globalObject = global;
this.setLastIndex(0); this.setLastIndex(0);
} }
NativeRegExp(final String input, final String flagString, final Global global) {
this(input, flagString, global, global.getRegExpPrototype());
}
NativeRegExp(final String input, final String flagString) { NativeRegExp(final String input, final String flagString) {
this(input, flagString, Global.instance()); this(input, flagString, Global.instance());
} }
@ -928,7 +932,7 @@ public final class NativeRegExp extends ScriptObject {
if (self instanceof NativeRegExp) { if (self instanceof NativeRegExp) {
return (NativeRegExp)self; return (NativeRegExp)self;
} else if (self != null && self == Global.instance().getRegExpPrototype()) { } else if (self != null && self == Global.instance().getRegExpPrototype()) {
return Global.instance().DEFAULT_REGEXP; return Global.instance().getDefaultRegExp();
} else { } else {
throw typeError("not.a.regexp", ScriptRuntime.safeToString(self)); throw typeError("not.a.regexp", ScriptRuntime.safeToString(self));
} }