Index: test/webkit/fast/js/caller-property.js |
diff --git a/test/webkit/fast/js/caller-property.js b/test/webkit/fast/js/caller-property.js |
new file mode 100644 |
index 0000000000000000000000000000000000000000..d28ccf8c504ce67b6c02d73d9633fdfcf5fe585d |
--- /dev/null |
+++ b/test/webkit/fast/js/caller-property.js |
@@ -0,0 +1,85 @@ |
+// Copyright 2013 the V8 project authors. All rights reserved. |
+// Copyright (C) 2005, 2006, 2007, 2008, 2009 Apple Inc. All rights reserved. |
+// |
+// Redistribution and use in source and binary forms, with or without |
+// modification, are permitted provided that the following conditions |
+// are met: |
+// 1. Redistributions of source code must retain the above copyright |
+// notice, this list of conditions and the following disclaimer. |
+// 2. 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. |
+// |
+// THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS 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 APPLE INC. OR ITS 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. |
+ |
+description( |
+'This tests for caller property in functions. Only functions that are called from inside of other functions and have a parent should have this property set. Tests return true when caller is found and false when the caller is null.' |
+) |
+function child() |
+{ |
+ return (child.caller !== null); |
+} |
+ |
+function parent() |
+{ |
+ return child(); |
+} |
+ |
+var childHasCallerWhenExecutingGlobalCode = (child.caller !== null); |
+var childHasCallerWhenCalledWithoutParent = child(); |
+var childHasCallerWhenCalledFromWithinParent = parent(); |
+ |
+shouldBe('childHasCallerWhenExecutingGlobalCode', 'false'); |
+shouldBe('childHasCallerWhenCalledWithoutParent', 'false'); |
+shouldBe('childHasCallerWhenCalledFromWithinParent', 'true') |
+ |
+// The caller property should throw in strict mode, and a non-strict function cannot use caller to reach a strict caller (see ES5.1 15.3.5.4). |
+function nonStrictCallee() { return nonStrictCallee.caller; } |
+function strictCallee() { "use strict"; return strictCallee.caller; } |
+function nonStrictCaller(x) { return x(); } |
+function strictCaller(x) { "use strict"; return x(); } |
+shouldBe("nonStrictCaller(nonStrictCallee)", "nonStrictCaller"); |
+shouldThrow("nonStrictCaller(strictCallee)", '"TypeError: Type error"'); |
+shouldThrow("strictCaller(nonStrictCallee)", '"TypeError: Function.caller used to retrieve strict caller"'); |
+shouldThrow("strictCaller(strictCallee)", '"TypeError: Type error"'); |
+ |
+// .caller within a bound function reaches the caller, ignoring the binding. |
+var boundNonStrictCallee = nonStrictCallee.bind(); |
+var boundStrictCallee = strictCallee.bind(); |
+shouldBe("nonStrictCaller(boundNonStrictCallee)", "nonStrictCaller"); |
+shouldThrow("nonStrictCaller(boundStrictCallee)", '"TypeError: Type error"'); |
+shouldThrow("strictCaller(boundNonStrictCallee)", '"TypeError: Function.caller used to retrieve strict caller"'); |
+shouldThrow("strictCaller(boundStrictCallee)", '"TypeError: Type error"'); |
+ |
+// Check that .caller works (or throws) as expected, over an accessor call. |
+function getFooGetter(x) { return Object.getOwnPropertyDescriptor(x, 'foo').get; } |
+function getFooSetter(x) { return Object.getOwnPropertyDescriptor(x, 'foo').set; } |
+var nonStrictAccessor = { |
+ get foo() { return getFooGetter(nonStrictAccessor).caller; }, |
+ set foo(x) { if (getFooSetter(nonStrictAccessor).caller !==x) throw false; } |
+}; |
+var strictAccessor = { |
+ get foo() { "use strict"; return getFooGetter(strictAccessor).caller; }, |
+ set foo(x) { "use strict"; if (getFooSetter(strictAccessor).caller !==x) throw false; } |
+}; |
+function nonStrictGetter(x) { return x.foo; } |
+function nonStrictSetter(x) { x.foo = nonStrictSetter; return true; } |
+function strictGetter(x) { "use strict"; return x.foo; } |
+function strictSetter(x) { "use strict"; x.foo = nonStrictSetter; return true; } |
+shouldBe("nonStrictGetter(nonStrictAccessor)", "nonStrictGetter"); |
+shouldBeTrue("nonStrictSetter(nonStrictAccessor)"); |
+shouldThrow("nonStrictGetter(strictAccessor)", '"TypeError: Type error"'); |
+shouldThrow("nonStrictSetter(strictAccessor)", '"TypeError: Type error"'); |
+shouldThrow("strictGetter(nonStrictAccessor)", '"TypeError: Function.caller used to retrieve strict caller"'); |
+shouldThrow("strictSetter(nonStrictAccessor)", '"TypeError: Function.caller used to retrieve strict caller"'); |
+shouldThrow("strictGetter(strictAccessor)", '"TypeError: Type error"'); |
+shouldThrow("strictSetter(strictAccessor)", '"TypeError: Type error"'); |