e164e6a4539d132924ee65bbdb7038619f5ba59c
[WebKit-https.git] / LayoutTests / js / script-tests / caller-property.js
1 description(
2 '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.'
3 )       
4 function child()
5 {
6     return (child.caller !== null);
7 }
8
9 function parent()
10 {
11     return child();
12 }
13
14 var childHasCallerWhenExecutingGlobalCode = (child.caller !== null);
15 var childHasCallerWhenCalledWithoutParent = child();
16 var childHasCallerWhenCalledFromWithinParent = parent();
17
18 shouldBe('childHasCallerWhenExecutingGlobalCode', 'false');
19 shouldBe('childHasCallerWhenCalledWithoutParent', 'false');
20 shouldBe('childHasCallerWhenCalledFromWithinParent', 'true')
21
22 // 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).
23 function nonStrictCallee() { return nonStrictCallee.caller; }
24 function strictCallee() { "use strict"; return strictCallee.caller; }
25 function nonStrictCaller(x) { return x(); }
26 // Tail calls leak and show our caller's caller, which is null here
27 function strictCaller(x) { "use strict"; var result = x(); return result; }
28 function strictTailCaller(x) { "use strict"; return x(); }
29 shouldBe("nonStrictCaller(nonStrictCallee)", "nonStrictCaller");
30 shouldThrow("nonStrictCaller(strictCallee)", '"TypeError: \'arguments\', \'callee\', and \'caller\' cannot be accessed in strict mode."');
31 shouldThrow("strictCaller(nonStrictCallee)");
32 shouldThrow("strictCaller(strictCallee)", '"TypeError: \'arguments\', \'callee\', and \'caller\' cannot be accessed in strict mode."');
33 shouldBe("strictTailCaller(nonStrictCallee)", "null");
34 shouldThrow("strictTailCaller(strictCallee)");
35
36 // .caller within a bound function reaches the caller, ignoring the binding.
37 var boundNonStrictCallee = nonStrictCallee.bind();
38 var boundStrictCallee = strictCallee.bind();
39 shouldBe("nonStrictCaller(boundNonStrictCallee)", "nonStrictCaller");
40 shouldThrow("nonStrictCaller(boundStrictCallee)");
41 shouldThrow("strictCaller(boundNonStrictCallee)", '"TypeError: Function.caller used to retrieve strict caller"');
42 shouldThrow("strictCaller(boundStrictCallee)");
43 shouldBe("strictTailCaller(boundNonStrictCallee)", "null");
44 shouldThrow("strictTailCaller(boundStrictCallee)");
45
46 // Check that .caller works (or throws) as expected, over an accessor call.
47 function getFooGetter(x) { return Object.getOwnPropertyDescriptor(x, 'foo').get; }
48 function getFooSetter(x) { return Object.getOwnPropertyDescriptor(x, 'foo').set; }
49 var nonStrictAccessor = {
50     get foo() { return getFooGetter(nonStrictAccessor).caller; },
51     set foo(x) { if (getFooSetter(nonStrictAccessor).caller !==x) throw false; }
52 };
53 var strictAccessor = {
54     get foo() { "use strict"; return getFooGetter(strictAccessor).caller; },
55     set foo(x) { "use strict"; if (getFooSetter(strictAccessor).caller !==x) throw false; }
56 };
57 function nonStrictGetter(x) { return x.foo; }
58 function nonStrictSetter(x) { x.foo = nonStrictSetter; return true; }
59 function strictGetter(x) { "use strict"; return x.foo; }
60 function strictSetter(x) { "use strict"; x.foo = nonStrictSetter; return true; }
61 shouldBe("nonStrictGetter(nonStrictAccessor)", "nonStrictGetter");
62 shouldBeTrue("nonStrictSetter(nonStrictAccessor)");
63 shouldThrow("nonStrictGetter(strictAccessor)");
64 shouldThrow("nonStrictSetter(strictAccessor)");
65 shouldThrow("strictGetter(nonStrictAccessor)", '"TypeError: Function.caller used to retrieve strict caller"');
66 shouldThrow("strictSetter(nonStrictAccessor)", '"TypeError: Function.caller used to retrieve strict caller"');
67 shouldThrow("strictGetter(strictAccessor)");
68 shouldThrow("strictSetter(strictAccessor)");