00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012
00013
00014
00015
00016
00017
00018
00019
00020
00021
00022
00023
00024
00025
00026
00027
00028 #include "v8.h"
00029
00030 #include "api.h"
00031 #include "debug.h"
00032 #include "execution.h"
00033 #include "factory.h"
00034 #include "macro-assembler.h"
00035
00036 namespace v8 { namespace internal {
00037
00038
00039 Handle<FixedArray> Factory::NewFixedArray(int size, PretenureFlag pretenure) {
00040 ASSERT(0 <= size);
00041 CALL_HEAP_FUNCTION(Heap::AllocateFixedArray(size, pretenure), FixedArray);
00042 }
00043
00044
00045 Handle<FixedArray> Factory::NewFixedArrayWithHoles(int size) {
00046 ASSERT(0 <= size);
00047 CALL_HEAP_FUNCTION(Heap::AllocateFixedArrayWithHoles(size), FixedArray);
00048 }
00049
00050
00051 Handle<Dictionary> Factory::NewDictionary(int at_least_space_for) {
00052 ASSERT(0 <= at_least_space_for);
00053 CALL_HEAP_FUNCTION(Dictionary::Allocate(at_least_space_for), Dictionary);
00054 }
00055
00056
00057 Handle<DescriptorArray> Factory::NewDescriptorArray(int number_of_descriptors) {
00058 ASSERT(0 <= number_of_descriptors);
00059 CALL_HEAP_FUNCTION(DescriptorArray::Allocate(number_of_descriptors),
00060 DescriptorArray);
00061 }
00062
00063
00064
00065 Handle<String> Factory::LookupSymbol(Vector<const char> string) {
00066 CALL_HEAP_FUNCTION(Heap::LookupSymbol(string), String);
00067 }
00068
00069
00070 Handle<String> Factory::NewStringFromAscii(Vector<const char> string,
00071 PretenureFlag pretenure) {
00072 CALL_HEAP_FUNCTION(Heap::AllocateStringFromAscii(string, pretenure), String);
00073 }
00074
00075 Handle<String> Factory::NewStringFromUtf8(Vector<const char> string,
00076 PretenureFlag pretenure) {
00077 CALL_HEAP_FUNCTION(Heap::AllocateStringFromUtf8(string, pretenure), String);
00078 }
00079
00080
00081 Handle<String> Factory::NewStringFromTwoByte(Vector<const uc16> string) {
00082 CALL_HEAP_FUNCTION(Heap::AllocateStringFromTwoByte(string), String);
00083 }
00084
00085
00086 Handle<String> Factory::NewRawTwoByteString(int length,
00087 PretenureFlag pretenure) {
00088 CALL_HEAP_FUNCTION(Heap::AllocateRawTwoByteString(length, pretenure), String);
00089 }
00090
00091
00092 Handle<String> Factory::NewConsString(Handle<String> first,
00093 Handle<String> second) {
00094 if (first->length() == 0) return second;
00095 if (second->length() == 0) return first;
00096 CALL_HEAP_FUNCTION(Heap::AllocateConsString(*first, *second), String);
00097 }
00098
00099
00100 Handle<String> Factory::NewStringSlice(Handle<String> str, int begin, int end) {
00101 CALL_HEAP_FUNCTION(str->Slice(begin, end), String);
00102 }
00103
00104
00105 Handle<String> Factory::NewExternalStringFromAscii(
00106 ExternalAsciiString::Resource* resource) {
00107 CALL_HEAP_FUNCTION(Heap::AllocateExternalStringFromAscii(resource), String);
00108 }
00109
00110
00111 Handle<String> Factory::NewExternalStringFromTwoByte(
00112 ExternalTwoByteString::Resource* resource) {
00113 CALL_HEAP_FUNCTION(Heap::AllocateExternalStringFromTwoByte(resource), String);
00114 }
00115
00116
00117 Handle<Context> Factory::NewGlobalContext() {
00118 CALL_HEAP_FUNCTION(Heap::AllocateGlobalContext(), Context);
00119 }
00120
00121
00122 Handle<Context> Factory::NewFunctionContext(int length,
00123 Handle<JSFunction> closure) {
00124 CALL_HEAP_FUNCTION(Heap::AllocateFunctionContext(length, *closure), Context);
00125 }
00126
00127
00128 Handle<Context> Factory::NewWithContext(Handle<Context> previous,
00129 Handle<JSObject> extension) {
00130 CALL_HEAP_FUNCTION(Heap::AllocateWithContext(*previous, *extension), Context);
00131 }
00132
00133
00134 Handle<Struct> Factory::NewStruct(InstanceType type) {
00135 CALL_HEAP_FUNCTION(Heap::AllocateStruct(type), Struct);
00136 }
00137
00138
00139 Handle<AccessorInfo> Factory::NewAccessorInfo() {
00140 Handle<AccessorInfo> info =
00141 Handle<AccessorInfo>::cast(NewStruct(ACCESSOR_INFO_TYPE));
00142 info->set_flag(0);
00143 return info;
00144 }
00145
00146
00147 Handle<Script> Factory::NewScript(Handle<String> source) {
00148 Handle<Script> script = Handle<Script>::cast(NewStruct(SCRIPT_TYPE));
00149 script->set_source(*source);
00150 script->set_name(Heap::undefined_value());
00151 script->set_line_offset(Smi::FromInt(0));
00152 script->set_column_offset(Smi::FromInt(0));
00153 script->set_type(Smi::FromInt(SCRIPT_TYPE_NORMAL));
00154 script->set_wrapper(*Factory::NewProxy(0, TENURED));
00155 return script;
00156 }
00157
00158
00159 Handle<Proxy> Factory::NewProxy(Address addr, PretenureFlag pretenure) {
00160 CALL_HEAP_FUNCTION(Heap::AllocateProxy(addr, pretenure), Proxy);
00161 }
00162
00163
00164 Handle<Proxy> Factory::NewProxy(const AccessorDescriptor* desc) {
00165 return NewProxy((Address) desc, TENURED);
00166 }
00167
00168
00169 Handle<ByteArray> Factory::NewByteArray(int length) {
00170 ASSERT(0 <= length);
00171 CALL_HEAP_FUNCTION(Heap::AllocateByteArray(length), ByteArray);
00172 }
00173
00174
00175 Handle<Map> Factory::NewMap(InstanceType type, int instance_size) {
00176 CALL_HEAP_FUNCTION(Heap::AllocateMap(type, instance_size), Map);
00177 }
00178
00179
00180 Handle<JSObject> Factory::NewFunctionPrototype(Handle<JSFunction> function) {
00181 CALL_HEAP_FUNCTION(Heap::AllocateFunctionPrototype(*function), JSObject);
00182 }
00183
00184
00185 Handle<Map> Factory::CopyMap(Handle<Map> src) {
00186 CALL_HEAP_FUNCTION(src->Copy(), Map);
00187 }
00188
00189
00190 Handle<Map> Factory::CopyMapDropTransitions(Handle<Map> src) {
00191 CALL_HEAP_FUNCTION(src->CopyDropTransitions(), Map);
00192 }
00193
00194
00195 Handle<FixedArray> Factory::CopyFixedArray(Handle<FixedArray> array) {
00196 CALL_HEAP_FUNCTION(array->Copy(), FixedArray);
00197 }
00198
00199
00200 Handle<JSFunction> Factory::BaseNewFunctionFromBoilerplate(
00201 Handle<JSFunction> boilerplate,
00202 Handle<Map> function_map) {
00203 ASSERT(boilerplate->IsBoilerplate());
00204 ASSERT(!boilerplate->has_initial_map());
00205 ASSERT(!boilerplate->has_prototype());
00206 ASSERT(boilerplate->properties() == Heap::empty_fixed_array());
00207 ASSERT(boilerplate->elements() == Heap::empty_fixed_array());
00208 CALL_HEAP_FUNCTION(Heap::AllocateFunction(*function_map,
00209 boilerplate->shared(),
00210 Heap::the_hole_value()),
00211 JSFunction);
00212 }
00213
00214
00215 Handle<JSFunction> Factory::NewFunctionFromBoilerplate(
00216 Handle<JSFunction> boilerplate,
00217 Handle<Context> context) {
00218 Handle<JSFunction> result =
00219 BaseNewFunctionFromBoilerplate(boilerplate, Top::function_map());
00220 result->set_context(*context);
00221 int number_of_literals = boilerplate->NumberOfLiterals();
00222 Handle<FixedArray> literals =
00223 Factory::NewFixedArray(number_of_literals, TENURED);
00224 if (number_of_literals > 0) {
00225
00226
00227
00228 literals->set(JSFunction::kLiteralGlobalContextIndex,
00229 context->global_context());
00230 }
00231 result->set_literals(*literals);
00232 ASSERT(!result->IsBoilerplate());
00233 return result;
00234 }
00235
00236
00237 Handle<Object> Factory::NewNumber(double value,
00238 PretenureFlag pretenure) {
00239 CALL_HEAP_FUNCTION(Heap::NumberFromDouble(value, pretenure), Object);
00240 }
00241
00242
00243 Handle<Object> Factory::NewNumberFromInt(int value) {
00244 CALL_HEAP_FUNCTION(Heap::NumberFromInt32(value), Object);
00245 }
00246
00247
00248 Handle<JSObject> Factory::NewNeanderObject() {
00249 CALL_HEAP_FUNCTION(Heap::AllocateJSObjectFromMap(Heap::neander_map()),
00250 JSObject);
00251 }
00252
00253
00254 Handle<Object> Factory::NewTypeError(const char* type,
00255 Vector< Handle<Object> > args) {
00256 return NewError("MakeTypeError", type, args);
00257 }
00258
00259
00260 Handle<Object> Factory::NewTypeError(Handle<String> message) {
00261 return NewError("$TypeError", message);
00262 }
00263
00264
00265 Handle<Object> Factory::NewRangeError(const char* type,
00266 Vector< Handle<Object> > args) {
00267 return NewError("MakeRangeError", type, args);
00268 }
00269
00270
00271 Handle<Object> Factory::NewRangeError(Handle<String> message) {
00272 return NewError("$RangeError", message);
00273 }
00274
00275
00276 Handle<Object> Factory::NewSyntaxError(const char* type, Handle<JSArray> args) {
00277 return NewError("MakeSyntaxError", type, args);
00278 }
00279
00280
00281 Handle<Object> Factory::NewSyntaxError(Handle<String> message) {
00282 return NewError("$SyntaxError", message);
00283 }
00284
00285
00286 Handle<Object> Factory::NewReferenceError(const char* type,
00287 Vector< Handle<Object> > args) {
00288 return NewError("MakeReferenceError", type, args);
00289 }
00290
00291
00292 Handle<Object> Factory::NewReferenceError(Handle<String> message) {
00293 return NewError("$ReferenceError", message);
00294 }
00295
00296
00297 Handle<Object> Factory::NewError(const char* maker, const char* type,
00298 Vector< Handle<Object> > args) {
00299 HandleScope scope;
00300 Handle<FixedArray> array = Factory::NewFixedArray(args.length());
00301 for (int i = 0; i < args.length(); i++) {
00302 array->set(i, *args[i]);
00303 }
00304 Handle<JSArray> object = Factory::NewJSArrayWithElements(array);
00305 Handle<Object> result = NewError(maker, type, object);
00306 return result.EscapeFrom(&scope);
00307 }
00308
00309
00310 Handle<Object> Factory::NewEvalError(const char* type,
00311 Vector< Handle<Object> > args) {
00312 return NewError("MakeEvalError", type, args);
00313 }
00314
00315
00316 Handle<Object> Factory::NewError(const char* type,
00317 Vector< Handle<Object> > args) {
00318 return NewError("MakeError", type, args);
00319 }
00320
00321
00322 Handle<Object> Factory::NewError(const char* maker,
00323 const char* type,
00324 Handle<JSArray> args) {
00325 Handle<String> make_str = Factory::LookupAsciiSymbol(maker);
00326 Handle<JSFunction> fun =
00327 Handle<JSFunction>(
00328 JSFunction::cast(
00329 Top::builtins()->GetProperty(*make_str)));
00330 Handle<Object> type_obj = Factory::LookupAsciiSymbol(type);
00331 Object** argv[2] = { type_obj.location(),
00332 Handle<Object>::cast(args).location() };
00333
00334
00335
00336 bool caught_exception;
00337 Handle<Object> result = Execution::TryCall(fun,
00338 Top::builtins(),
00339 2,
00340 argv,
00341 &caught_exception);
00342 return result;
00343 }
00344
00345
00346 Handle<Object> Factory::NewError(Handle<String> message) {
00347 return NewError("$Error", message);
00348 }
00349
00350
00351 Handle<Object> Factory::NewError(const char* constructor,
00352 Handle<String> message) {
00353 Handle<String> constr = Factory::LookupAsciiSymbol(constructor);
00354 Handle<JSFunction> fun =
00355 Handle<JSFunction>(
00356 JSFunction::cast(
00357 Top::builtins()->GetProperty(*constr)));
00358 Object** argv[1] = { Handle<Object>::cast(message).location() };
00359
00360
00361
00362 bool caught_exception;
00363 Handle<Object> result = Execution::TryCall(fun,
00364 Top::builtins(),
00365 1,
00366 argv,
00367 &caught_exception);
00368 return result;
00369 }
00370
00371
00372 Handle<JSFunction> Factory::NewFunction(Handle<String> name,
00373 InstanceType type,
00374 int instance_size,
00375 Handle<Code> code,
00376 bool force_initial_map) {
00377
00378 Handle<JSFunction> function = NewFunction(name, the_hole_value());
00379 function->set_code(*code);
00380
00381 if (force_initial_map ||
00382 type != JS_OBJECT_TYPE ||
00383 instance_size != JSObject::kHeaderSize) {
00384 Handle<Map> initial_map = NewMap(type, instance_size);
00385 Handle<JSObject> prototype = NewFunctionPrototype(function);
00386 initial_map->set_prototype(*prototype);
00387 function->set_initial_map(*initial_map);
00388 initial_map->set_constructor(*function);
00389 } else {
00390 ASSERT(!function->has_initial_map());
00391 ASSERT(!function->has_prototype());
00392 }
00393
00394 return function;
00395 }
00396
00397
00398 Handle<JSFunction> Factory::NewFunctionBoilerplate(Handle<String> name,
00399 int number_of_literals,
00400 bool contains_array_literal,
00401 Handle<Code> code) {
00402 Handle<JSFunction> function = NewFunctionBoilerplate(name);
00403 function->set_code(*code);
00404 int literals_array_size = number_of_literals;
00405
00406
00407
00408 if (number_of_literals > 0 || contains_array_literal) {
00409 literals_array_size += JSFunction::kLiteralsPrefixSize;
00410 }
00411 Handle<FixedArray> literals =
00412 Factory::NewFixedArray(literals_array_size, TENURED);
00413 function->set_literals(*literals);
00414 ASSERT(!function->has_initial_map());
00415 ASSERT(!function->has_prototype());
00416 return function;
00417 }
00418
00419
00420 Handle<JSFunction> Factory::NewFunctionBoilerplate(Handle<String> name) {
00421 Handle<SharedFunctionInfo> shared = NewSharedFunctionInfo(name);
00422 CALL_HEAP_FUNCTION(Heap::AllocateFunction(Heap::boilerplate_function_map(),
00423 *shared,
00424 Heap::the_hole_value()),
00425 JSFunction);
00426 }
00427
00428
00429 Handle<JSFunction> Factory::NewFunctionWithPrototype(Handle<String> name,
00430 InstanceType type,
00431 int instance_size,
00432 Handle<JSObject> prototype,
00433 Handle<Code> code,
00434 bool force_initial_map) {
00435
00436 Handle<JSFunction> function = NewFunction(name, prototype);
00437
00438 function->set_code(*code);
00439
00440 if (force_initial_map ||
00441 type != JS_OBJECT_TYPE ||
00442 instance_size != JSObject::kHeaderSize) {
00443 Handle<Map> initial_map = NewMap(type, instance_size);
00444 function->set_initial_map(*initial_map);
00445 initial_map->set_constructor(*function);
00446 }
00447
00448
00449
00450 SetPrototypeProperty(function, prototype);
00451 SetProperty(prototype, Factory::constructor_symbol(), function, DONT_ENUM);
00452 return function;
00453 }
00454
00455
00456 Handle<Code> Factory::NewCode(const CodeDesc& desc, ScopeInfo<>* sinfo,
00457 Code::Flags flags) {
00458 CALL_HEAP_FUNCTION(Heap::CreateCode(desc, sinfo, flags), Code);
00459 }
00460
00461
00462 Handle<Code> Factory::CopyCode(Handle<Code> code) {
00463 CALL_HEAP_FUNCTION(Heap::CopyCode(*code), Code);
00464 }
00465
00466
00467 static inline Object* DoCopyInsert(DescriptorArray* array,
00468 String* key,
00469 Object* value,
00470 PropertyAttributes attributes) {
00471 CallbacksDescriptor desc(key, value, attributes);
00472 Object* obj = array->CopyInsert(&desc, REMOVE_TRANSITIONS);
00473 return obj;
00474 }
00475
00476
00477
00478 Handle<DescriptorArray> Factory::CopyAppendProxyDescriptor(
00479 Handle<DescriptorArray> array,
00480 Handle<String> key,
00481 Handle<Object> value,
00482 PropertyAttributes attributes) {
00483 CALL_HEAP_FUNCTION(DoCopyInsert(*array, *key, *value, attributes),
00484 DescriptorArray);
00485 }
00486
00487
00488 Handle<String> Factory::SymbolFromString(Handle<String> value) {
00489 CALL_HEAP_FUNCTION(Heap::LookupSymbol(*value), String);
00490 }
00491
00492
00493 Handle<DescriptorArray> Factory::CopyAppendCallbackDescriptors(
00494 Handle<DescriptorArray> array,
00495 Handle<Object> descriptors) {
00496 v8::NeanderArray callbacks(descriptors);
00497 int nof_callbacks = callbacks.length();
00498 Handle<DescriptorArray> result =
00499 NewDescriptorArray(array->number_of_descriptors() + nof_callbacks);
00500
00501
00502 int descriptor_count = 0;
00503
00504
00505 DescriptorWriter w(*result);
00506 for (DescriptorReader r(*array); !r.eos(); r.advance()) {
00507 w.WriteFrom(&r);
00508 descriptor_count++;
00509 }
00510
00511
00512 int duplicates = 0;
00513
00514
00515
00516
00517 for (int i = nof_callbacks - 1; i >= 0; i--) {
00518 Handle<AccessorInfo> entry =
00519 Handle<AccessorInfo>(AccessorInfo::cast(callbacks.get(i)));
00520
00521 Handle<String> key =
00522 SymbolFromString(Handle<String>(String::cast(entry->name())));
00523
00524 if (result->LinearSearch(*key, descriptor_count) ==
00525 DescriptorArray::kNotFound) {
00526 CallbacksDescriptor desc(*key, *entry, entry->property_attributes());
00527 w.Write(&desc);
00528 descriptor_count++;
00529 } else {
00530 duplicates++;
00531 }
00532 }
00533
00534
00535
00536 if (duplicates > 0) {
00537 Handle<DescriptorArray> new_result =
00538 NewDescriptorArray(result->number_of_descriptors() - duplicates);
00539 DescriptorWriter w(*new_result);
00540 DescriptorReader r(*result);
00541 while (!w.eos()) {
00542 w.WriteFrom(&r);
00543 r.advance();
00544 }
00545 result = new_result;
00546 }
00547
00548
00549 result->Sort();
00550 return result;
00551 }
00552
00553
00554 Handle<JSObject> Factory::NewJSObject(Handle<JSFunction> constructor,
00555 PretenureFlag pretenure) {
00556 CALL_HEAP_FUNCTION(Heap::AllocateJSObject(*constructor, pretenure), JSObject);
00557 }
00558
00559
00560 Handle<JSObject> Factory::NewJSObjectFromMap(Handle<Map> map) {
00561 CALL_HEAP_FUNCTION(Heap::AllocateJSObjectFromMap(*map, NOT_TENURED),
00562 JSObject);
00563 }
00564
00565
00566 Handle<JSObject> Factory::NewObjectLiteral(int expected_number_of_properties) {
00567 Handle<Map> map = Handle<Map>(Top::object_function()->initial_map());
00568 map = Factory::CopyMap(map);
00569 map->set_instance_descriptors(Heap::empty_descriptor_array());
00570 map->set_unused_property_fields(expected_number_of_properties);
00571 CALL_HEAP_FUNCTION(Heap::AllocateJSObjectFromMap(*map, TENURED),
00572 JSObject);
00573 }
00574
00575
00576 Handle<JSArray> Factory::NewArrayLiteral(int length) {
00577 return NewJSArrayWithElements(NewFixedArray(length), TENURED);
00578 }
00579
00580
00581 Handle<JSArray> Factory::NewJSArray(int length,
00582 PretenureFlag pretenure) {
00583 Handle<JSObject> obj = NewJSObject(Top::array_function(), pretenure);
00584 CALL_HEAP_FUNCTION(Handle<JSArray>::cast(obj)->Initialize(length), JSArray);
00585 }
00586
00587
00588 Handle<JSArray> Factory::NewJSArrayWithElements(Handle<FixedArray> elements,
00589 PretenureFlag pretenure) {
00590 Handle<JSArray> result =
00591 Handle<JSArray>::cast(NewJSObject(Top::array_function(), pretenure));
00592 result->SetContent(*elements);
00593 return result;
00594 }
00595
00596
00597 Handle<SharedFunctionInfo> Factory::NewSharedFunctionInfo(Handle<String> name) {
00598 CALL_HEAP_FUNCTION(Heap::AllocateSharedFunctionInfo(*name),
00599 SharedFunctionInfo);
00600 }
00601
00602
00603 Handle<Dictionary> Factory::DictionaryAtNumberPut(Handle<Dictionary> dictionary,
00604 uint32_t key,
00605 Handle<Object> value) {
00606 CALL_HEAP_FUNCTION(dictionary->AtNumberPut(key, *value), Dictionary);
00607 }
00608
00609
00610 Handle<JSFunction> Factory::NewFunctionHelper(Handle<String> name,
00611 Handle<Object> prototype) {
00612 Handle<SharedFunctionInfo> function_share = NewSharedFunctionInfo(name);
00613 CALL_HEAP_FUNCTION(Heap::AllocateFunction(*Top::function_map(),
00614 *function_share,
00615 *prototype),
00616 JSFunction);
00617 }
00618
00619
00620 Handle<JSFunction> Factory::NewFunction(Handle<String> name,
00621 Handle<Object> prototype) {
00622 Handle<JSFunction> fun = NewFunctionHelper(name, prototype);
00623 fun->set_context(Top::context()->global_context());
00624 return fun;
00625 }
00626
00627
00628 Handle<Object> Factory::ToObject(Handle<Object> object,
00629 Handle<Context> global_context) {
00630 CALL_HEAP_FUNCTION(object->ToObject(*global_context), Object);
00631 }
00632
00633
00634 Handle<DebugInfo> Factory::NewDebugInfo(Handle<SharedFunctionInfo> shared) {
00635
00636 Handle<Code> code(shared->code());
00637
00638
00639
00640 Handle<Code> original_code(*Factory::CopyCode(code));
00641
00642
00643
00644
00645 Handle<FixedArray> break_points(
00646 Factory::NewFixedArray(Debug::kEstimatedNofBreakPointsInFunction));
00647
00648
00649
00650
00651 Handle<DebugInfo> debug_info =
00652 Handle<DebugInfo>::cast(Factory::NewStruct(DEBUG_INFO_TYPE));
00653 debug_info->set_shared(*shared);
00654 debug_info->set_original_code(*original_code);
00655 debug_info->set_code(*code);
00656 debug_info->set_break_points(*break_points);
00657
00658
00659 shared->set_debug_info(*debug_info);
00660
00661 return debug_info;
00662 }
00663
00664
00665 Handle<JSObject> Factory::NewArgumentsObject(Handle<Object> callee,
00666 int length) {
00667 CALL_HEAP_FUNCTION(Heap::AllocateArgumentsObject(*callee, length), JSObject);
00668 }
00669
00670
00671 Handle<JSFunction> Factory::CreateApiFunction(
00672 Handle<FunctionTemplateInfo> obj, ApiInstanceType instance_type) {
00673 Handle<Code> code = Handle<Code>(Builtins::builtin(Builtins::HandleApiCall));
00674
00675 int internal_field_count = 0;
00676 if (!obj->instance_template()->IsUndefined()) {
00677 Handle<ObjectTemplateInfo> instance_template =
00678 Handle<ObjectTemplateInfo>(
00679 ObjectTemplateInfo::cast(obj->instance_template()));
00680 internal_field_count =
00681 Smi::cast(instance_template->internal_field_count())->value();
00682 }
00683
00684 int instance_size = kPointerSize * internal_field_count;
00685 InstanceType type = INVALID_TYPE;
00686 switch (instance_type) {
00687 case JavaScriptObject:
00688 type = JS_OBJECT_TYPE;
00689 instance_size += JSObject::kHeaderSize;
00690 break;
00691 case InnerGlobalObject:
00692 type = JS_GLOBAL_OBJECT_TYPE;
00693 instance_size += JSGlobalObject::kSize;
00694 break;
00695 case OuterGlobalObject:
00696 type = JS_GLOBAL_PROXY_TYPE;
00697 instance_size += JSGlobalProxy::kSize;
00698 break;
00699 default:
00700 break;
00701 }
00702 ASSERT(type != INVALID_TYPE);
00703
00704 Handle<JSFunction> result =
00705 Factory::NewFunction(Factory::empty_symbol(), type, instance_size,
00706 code, true);
00707
00708 Handle<Object> class_name = Handle<Object>(obj->class_name());
00709 if (class_name->IsString()) {
00710 result->shared()->set_instance_class_name(*class_name);
00711 result->shared()->set_name(*class_name);
00712 }
00713
00714 Handle<Map> map = Handle<Map>(result->initial_map());
00715
00716
00717 if (obj->undetectable()) {
00718 map->set_is_undetectable();
00719 }
00720
00721
00722 if (obj->hidden_prototype()) {
00723 map->set_is_hidden_prototype();
00724 }
00725
00726
00727 if (obj->needs_access_check()) {
00728 map->set_is_access_check_needed();
00729 }
00730
00731
00732 if (!obj->named_property_handler()->IsUndefined()) {
00733 map->set_has_named_interceptor();
00734 }
00735 if (!obj->indexed_property_handler()->IsUndefined()) {
00736 map->set_has_indexed_interceptor();
00737 }
00738
00739
00740 if (!obj->instance_call_handler()->IsUndefined()) {
00741 map->set_has_instance_call_handler();
00742 }
00743
00744 result->shared()->set_function_data(*obj);
00745 result->shared()->DontAdaptArguments();
00746
00747
00748 Handle<DescriptorArray> array =
00749 Handle<DescriptorArray>(map->instance_descriptors());
00750 while (true) {
00751 Handle<Object> props = Handle<Object>(obj->property_accessors());
00752 if (!props->IsUndefined()) {
00753 array = Factory::CopyAppendCallbackDescriptors(array, props);
00754 }
00755 Handle<Object> parent = Handle<Object>(obj->parent_template());
00756 if (parent->IsUndefined()) break;
00757 obj = Handle<FunctionTemplateInfo>::cast(parent);
00758 }
00759 if (!array->IsEmpty()) {
00760 map->set_instance_descriptors(*array);
00761 }
00762
00763 return result;
00764 }
00765
00766
00767 Handle<MapCache> Factory::NewMapCache(int at_least_space_for) {
00768 CALL_HEAP_FUNCTION(MapCache::Allocate(at_least_space_for), MapCache);
00769 }
00770
00771
00772 static Object* UpdateMapCacheWith(Context* context,
00773 FixedArray* keys,
00774 Map* map) {
00775 Object* result = MapCache::cast(context->map_cache())->Put(keys, map);
00776 if (!result->IsFailure()) context->set_map_cache(MapCache::cast(result));
00777 return result;
00778 }
00779
00780
00781 Handle<MapCache> Factory::AddToMapCache(Handle<Context> context,
00782 Handle<FixedArray> keys,
00783 Handle<Map> map) {
00784 CALL_HEAP_FUNCTION(UpdateMapCacheWith(*context, *keys, *map), MapCache);
00785 }
00786
00787
00788 Handle<Map> Factory::ObjectLiteralMapFromCache(Handle<Context> context,
00789 Handle<FixedArray> keys) {
00790 if (context->map_cache()->IsUndefined()) {
00791
00792 Handle<MapCache> new_cache = NewMapCache(24);
00793 context->set_map_cache(*new_cache);
00794 }
00795
00796 Handle<MapCache> cache =
00797 Handle<MapCache>(MapCache::cast(context->map_cache()));
00798 Handle<Object> result = Handle<Object>(cache->Lookup(*keys));
00799 if (result->IsMap()) return Handle<Map>::cast(result);
00800
00801 Handle<Map> map =
00802 CopyMap(Handle<Map>(context->object_function()->initial_map()));
00803 AddToMapCache(context, keys, map);
00804 return Handle<Map>(map);
00805 }
00806
00807
00808 void Factory::SetRegExpData(Handle<JSRegExp> regexp,
00809 JSRegExp::Type type,
00810 Handle<String> source,
00811 JSRegExp::Flags flags,
00812 Handle<Object> data) {
00813 Handle<FixedArray> store = NewFixedArray(JSRegExp::kDataSize);
00814 store->set(JSRegExp::kTagIndex, Smi::FromInt(type));
00815 store->set(JSRegExp::kSourceIndex, *source);
00816 store->set(JSRegExp::kFlagsIndex, Smi::FromInt(flags.value()));
00817 store->set(JSRegExp::kAtomPatternIndex, *data);
00818 regexp->set_data(*store);
00819 }
00820
00821
00822 void Factory::ConfigureInstance(Handle<FunctionTemplateInfo> desc,
00823 Handle<JSObject> instance,
00824 bool* pending_exception) {
00825
00826
00827 Handle<Object> instance_template = Handle<Object>(desc->instance_template());
00828 if (!instance_template->IsUndefined()) {
00829 Execution::ConfigureInstance(instance,
00830 instance_template,
00831 pending_exception);
00832 } else {
00833 *pending_exception = false;
00834 }
00835 }
00836
00837
00838 } }