Add possibility to configure 'prototype' property via FunctionTemplate

BUG=v8:1479
TEST=test-api/SetPrototypeProperties

Review URL: http://codereview.chromium.org/7229007

git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@8366 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
parent e8b2016b
......@@ -2143,6 +2143,13 @@ class V8EXPORT FunctionTemplate : public Template {
*/
void SetHiddenPrototype(bool value);
/**
* Sets the property attributes of the 'prototype' property of functions
* created from this FunctionTemplate. Can be any combination of ReadOnly,
* DontEnum and DontDelete.
*/
void SetPrototypeAttributes(int attributes);
/**
* Returns true if the given object is an instance of this function
* template.
......
......@@ -884,6 +884,7 @@ static void InitializeFunctionTemplate(
i::Handle<i::FunctionTemplateInfo> info) {
info->set_tag(i::Smi::FromInt(Consts::FUNCTION_TEMPLATE));
info->set_flag(0);
info->set_prototype_attributes(i::Smi::FromInt(v8::None));
}
......@@ -1106,6 +1107,17 @@ void FunctionTemplate::SetHiddenPrototype(bool value) {
}
void FunctionTemplate::SetPrototypeAttributes(int attributes) {
i::Isolate* isolate = Utils::OpenHandle(this)->GetIsolate();
if (IsDeadCheck(isolate, "v8::FunctionTemplate::SetPrototypeAttributes()")) {
return;
}
ENTER_V8(isolate);
Utils::OpenHandle(this)->set_prototype_attributes(
i::Smi::FromInt(attributes));
}
void FunctionTemplate::SetNamedInstancePropertyHandler(
NamedPropertyGetter getter,
NamedPropertySetter setter,
......
......@@ -73,7 +73,15 @@ function InstantiateFunction(data, name) {
if (name) %FunctionSetName(fun, name);
cache[serialNumber] = fun;
var prototype = %GetTemplateField(data, kApiPrototypeTemplateOffset);
fun.prototype = prototype ? Instantiate(prototype) : {};
var attributes = %GetTemplateField(data, kApiPrototypeAttributesOffset);
if (attributes != NONE) {
%IgnoreAttributesAndSetProperty(
fun, "prototype",
prototype ? Instantiate(prototype) : {},
attributes);
} else {
fun.prototype = prototype ? Instantiate(prototype) : {};
}
%SetProperty(fun.prototype, "constructor", fun, DONT_ENUM);
var parent = %GetTemplateField(data, kApiParentTemplateOffset);
if (parent) {
......
......@@ -38,12 +38,13 @@ const GETTER = 0;
const SETTER = 1;
# These definitions must match the index of the properties in objects.h.
const kApiTagOffset = 0;
const kApiPropertyListOffset = 1;
const kApiSerialNumberOffset = 2;
const kApiConstructorOffset = 2;
const kApiPrototypeTemplateOffset = 5;
const kApiParentTemplateOffset = 6;
const kApiTagOffset = 0;
const kApiPropertyListOffset = 1;
const kApiSerialNumberOffset = 2;
const kApiConstructorOffset = 2;
const kApiPrototypeTemplateOffset = 5;
const kApiParentTemplateOffset = 6;
const kApiPrototypeAttributesOffset = 15;
const NO_HINT = 0;
const NUMBER_HINT = 1;
......
......@@ -3232,6 +3232,8 @@ ACCESSORS(FunctionTemplateInfo, instance_call_handler, Object,
ACCESSORS(FunctionTemplateInfo, access_check_info, Object,
kAccessCheckInfoOffset)
ACCESSORS(FunctionTemplateInfo, flag, Smi, kFlagOffset)
ACCESSORS(FunctionTemplateInfo, prototype_attributes, Smi,
kPrototypeAttributesOffset)
ACCESSORS(ObjectTemplateInfo, constructor, Object, kConstructorOffset)
ACCESSORS(ObjectTemplateInfo, internal_field_count, Object,
......
......@@ -6786,6 +6786,7 @@ class FunctionTemplateInfo: public TemplateInfo {
DECL_ACCESSORS(instance_call_handler, Object)
DECL_ACCESSORS(access_check_info, Object)
DECL_ACCESSORS(flag, Smi)
DECL_ACCESSORS(prototype_attributes, Smi)
// Following properties use flag bits.
DECL_BOOLEAN_ACCESSORS(hidden_prototype)
......@@ -6825,7 +6826,8 @@ class FunctionTemplateInfo: public TemplateInfo {
static const int kAccessCheckInfoOffset =
kInstanceCallHandlerOffset + kPointerSize;
static const int kFlagOffset = kAccessCheckInfoOffset + kPointerSize;
static const int kSize = kFlagOffset + kPointerSize;
static const int kPrototypeAttributesOffset = kFlagOffset + kPointerSize;
static const int kSize = kPrototypeAttributesOffset + kPointerSize;
private:
// Bit position in the flag, from least significant bit position.
......
......@@ -6842,6 +6842,56 @@ THREADED_TEST(SetPrototype) {
}
THREADED_TEST(SetPrototypeProperties) {
v8::HandleScope handle_scope;
LocalContext context;
Local<v8::FunctionTemplate> t1 = v8::FunctionTemplate::New();
t1->SetPrototypeAttributes(v8::DontDelete);
context->Global()->Set(v8_str("func1"), t1->GetFunction());
CHECK(CompileRun(
"(function() {"
" descriptor = Object.getOwnPropertyDescriptor(func1, 'prototype');"
" return (descriptor['writable'] == true) &&"
" (descriptor['enumerable'] == true) &&"
" (descriptor['configurable'] == false);"
"})()")->BooleanValue());
Local<v8::FunctionTemplate> t2 = v8::FunctionTemplate::New();
t2->SetPrototypeAttributes(v8::DontEnum);
context->Global()->Set(v8_str("func2"), t2->GetFunction());
CHECK(CompileRun(
"(function() {"
" descriptor = Object.getOwnPropertyDescriptor(func2, 'prototype');"
" return (descriptor['writable'] == true) &&"
" (descriptor['enumerable'] == false) &&"
" (descriptor['configurable'] == true);"
"})()")->BooleanValue());
Local<v8::FunctionTemplate> t3 = v8::FunctionTemplate::New();
t3->SetPrototypeAttributes(v8::ReadOnly);
context->Global()->Set(v8_str("func3"), t3->GetFunction());
CHECK(CompileRun(
"(function() {"
" descriptor = Object.getOwnPropertyDescriptor(func3, 'prototype');"
" return (descriptor['writable'] == false) &&"
" (descriptor['enumerable'] == true) &&"
" (descriptor['configurable'] == true);"
"})()")->BooleanValue());
Local<v8::FunctionTemplate> t4 = v8::FunctionTemplate::New();
t4->SetPrototypeAttributes(v8::ReadOnly | v8::DontEnum | v8::DontDelete);
context->Global()->Set(v8_str("func4"), t4->GetFunction());
CHECK(CompileRun(
"(function() {"
" descriptor = Object.getOwnPropertyDescriptor(func4, 'prototype');"
" return (descriptor['writable'] == false) &&"
" (descriptor['enumerable'] == false) &&"
" (descriptor['configurable'] == false);"
"})()")->BooleanValue());
}
THREADED_TEST(SetPrototypeThrows) {
v8::HandleScope handle_scope;
LocalContext context;
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment