| Index: test/cctest/test-api.cc
|
| diff --git a/test/cctest/test-api.cc b/test/cctest/test-api.cc
|
| index bd543ec481db0bd24cf4fd04e3aa7aec474a7578..e5e1e0d227a15892f1f7ef39b0de50402015202a 100644
|
| --- a/test/cctest/test-api.cc
|
| +++ b/test/cctest/test-api.cc
|
| @@ -4843,6 +4843,84 @@ THREADED_TEST(HiddenPrototype) {
|
| }
|
|
|
|
|
| +THREADED_TEST(SetPrototype) {
|
| + v8::HandleScope handle_scope;
|
| + LocalContext context;
|
| +
|
| + Local<v8::FunctionTemplate> t0 = v8::FunctionTemplate::New();
|
| + t0->InstanceTemplate()->Set(v8_str("x"), v8_num(0));
|
| + Local<v8::FunctionTemplate> t1 = v8::FunctionTemplate::New();
|
| + t1->SetHiddenPrototype(true);
|
| + t1->InstanceTemplate()->Set(v8_str("y"), v8_num(1));
|
| + Local<v8::FunctionTemplate> t2 = v8::FunctionTemplate::New();
|
| + t2->SetHiddenPrototype(true);
|
| + t2->InstanceTemplate()->Set(v8_str("z"), v8_num(2));
|
| + Local<v8::FunctionTemplate> t3 = v8::FunctionTemplate::New();
|
| + t3->InstanceTemplate()->Set(v8_str("u"), v8_num(3));
|
| +
|
| + Local<v8::Object> o0 = t0->GetFunction()->NewInstance();
|
| + Local<v8::Object> o1 = t1->GetFunction()->NewInstance();
|
| + Local<v8::Object> o2 = t2->GetFunction()->NewInstance();
|
| + Local<v8::Object> o3 = t3->GetFunction()->NewInstance();
|
| +
|
| + // Setting the prototype on an object does not skip hidden prototypes.
|
| + CHECK_EQ(0, o0->Get(v8_str("x"))->Int32Value());
|
| + CHECK(o0->SetPrototype(o1));
|
| + CHECK_EQ(0, o0->Get(v8_str("x"))->Int32Value());
|
| + CHECK_EQ(1, o0->Get(v8_str("y"))->Int32Value());
|
| + CHECK(o1->SetPrototype(o2));
|
| + CHECK_EQ(0, o0->Get(v8_str("x"))->Int32Value());
|
| + CHECK_EQ(1, o0->Get(v8_str("y"))->Int32Value());
|
| + CHECK_EQ(2, o0->Get(v8_str("z"))->Int32Value());
|
| + CHECK(o2->SetPrototype(o3));
|
| + CHECK_EQ(0, o0->Get(v8_str("x"))->Int32Value());
|
| + CHECK_EQ(1, o0->Get(v8_str("y"))->Int32Value());
|
| + CHECK_EQ(2, o0->Get(v8_str("z"))->Int32Value());
|
| + CHECK_EQ(3, o0->Get(v8_str("u"))->Int32Value());
|
| +
|
| + // Getting the prototype of o0 should get the first visible one
|
| + // which is o3. Therefore, z should not be defined on the prototype
|
| + // object.
|
| + Local<Value> proto = o0->Get(v8_str("__proto__"));
|
| + CHECK(proto->IsObject());
|
| + CHECK_EQ(v8::Handle<v8::Object>::Cast(proto), o3);
|
| +
|
| + // However, Object::GetPrototype ignores hidden prototype.
|
| + Local<Value> proto0 = o0->GetPrototype();
|
| + CHECK(proto0->IsObject());
|
| + CHECK_EQ(v8::Handle<v8::Object>::Cast(proto0), o1);
|
| +
|
| + Local<Value> proto1 = o1->GetPrototype();
|
| + CHECK(proto1->IsObject());
|
| + CHECK_EQ(v8::Handle<v8::Object>::Cast(proto1), o2);
|
| +
|
| + Local<Value> proto2 = o2->GetPrototype();
|
| + CHECK(proto2->IsObject());
|
| + CHECK_EQ(v8::Handle<v8::Object>::Cast(proto2), o3);
|
| +}
|
| +
|
| +
|
| +THREADED_TEST(SetPrototypeThrows) {
|
| + v8::HandleScope handle_scope;
|
| + LocalContext context;
|
| +
|
| + Local<v8::FunctionTemplate> t = v8::FunctionTemplate::New();
|
| +
|
| + Local<v8::Object> o0 = t->GetFunction()->NewInstance();
|
| + Local<v8::Object> o1 = t->GetFunction()->NewInstance();
|
| +
|
| + CHECK(o0->SetPrototype(o1));
|
| + // If setting the prototype leads to the cycle, SetPrototype should
|
| + // return false and keep VM in sane state.
|
| + v8::TryCatch try_catch;
|
| + CHECK(!o1->SetPrototype(o0));
|
| + CHECK(!try_catch.HasCaught());
|
| + ASSERT(!i::Top::has_pending_exception());
|
| +
|
| + CHECK_EQ(42, CompileRun("function f() { return 42; }; f()")->Int32Value());
|
| +}
|
| +
|
| +
|
| THREADED_TEST(GetterSetterExceptions) {
|
| v8::HandleScope handle_scope;
|
| LocalContext context;
|
|
|