// Copyright 2006-2008 the V8 project authors. All rights reserved. // Redistribution and use in source and binary forms, with or without // modification, are permitted provided that the following conditions are // met: // // * Redistributions of source code must retain the above copyright // notice, this list of conditions and the following disclaimer. // * 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. // * Neither the name of Google Inc. nor the names of its // contributors may be used to endorse or promote products derived // from this software without specific prior written permission. // // THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND 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 THE COPYRIGHT // OWNER OR 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. #include "v8.h" #include "accessors.h" #include "api.h" #include "bootstrapper.h" #include "compiler.h" #include "debug.h" #include "execution.h" #include "global-handles.h" #include "macro-assembler.h" #include "natives.h" #include "objects-visiting.h" #include "snapshot.h" #include "extensions/externalize-string-extension.h" #include "extensions/gc-extension.h" namespace v8 { namespace internal { NativesExternalStringResource::NativesExternalStringResource( Bootstrapper* bootstrapper, const char* source) : data_(source), length_(StrLength(source)) { if (bootstrapper->delete_these_non_arrays_on_tear_down_ == NULL) { bootstrapper->delete_these_non_arrays_on_tear_down_ = new List(2); } // The resources are small objects and we only make a fixed number of // them, but let's clean them up on exit for neatness. bootstrapper->delete_these_non_arrays_on_tear_down_-> Add(reinterpret_cast(this)); } Bootstrapper::Bootstrapper() : nesting_(0), extensions_cache_(Script::TYPE_EXTENSION), delete_these_non_arrays_on_tear_down_(NULL), delete_these_arrays_on_tear_down_(NULL) { } Handle Bootstrapper::NativesSourceLookup(int index) { ASSERT(0 <= index && index < Natives::GetBuiltinsCount()); Isolate* isolate = Isolate::Current(); Factory* factory = isolate->factory(); Heap* heap = isolate->heap(); if (heap->natives_source_cache()->get(index)->IsUndefined()) { if (!Snapshot::IsEnabled() || FLAG_new_snapshot) { // We can use external strings for the natives. NativesExternalStringResource* resource = new NativesExternalStringResource(this, Natives::GetScriptSource(index).start()); Handle source_code = factory->NewExternalStringFromAscii(resource); heap->natives_source_cache()->set(index, *source_code); } else { // Old snapshot code can't cope with external strings at all. Handle source_code = factory->NewStringFromAscii(Natives::GetScriptSource(index)); heap->natives_source_cache()->set(index, *source_code); } } Handle cached_source(heap->natives_source_cache()->get(index)); return Handle::cast(cached_source); } void Bootstrapper::Initialize(bool create_heap_objects) { extensions_cache_.Initialize(create_heap_objects); GCExtension::Register(); ExternalizeStringExtension::Register(); } char* Bootstrapper::AllocateAutoDeletedArray(int bytes) { char* memory = new char[bytes]; if (memory != NULL) { if (delete_these_arrays_on_tear_down_ == NULL) { delete_these_arrays_on_tear_down_ = new List(2); } delete_these_arrays_on_tear_down_->Add(memory); } return memory; } void Bootstrapper::TearDown() { if (delete_these_non_arrays_on_tear_down_ != NULL) { int len = delete_these_non_arrays_on_tear_down_->length(); ASSERT(len < 20); // Don't use this mechanism for unbounded allocations. for (int i = 0; i < len; i++) { delete delete_these_non_arrays_on_tear_down_->at(i); delete_these_non_arrays_on_tear_down_->at(i) = NULL; } delete delete_these_non_arrays_on_tear_down_; delete_these_non_arrays_on_tear_down_ = NULL; } if (delete_these_arrays_on_tear_down_ != NULL) { int len = delete_these_arrays_on_tear_down_->length(); ASSERT(len < 1000); // Don't use this mechanism for unbounded allocations. for (int i = 0; i < len; i++) { delete[] delete_these_arrays_on_tear_down_->at(i); delete_these_arrays_on_tear_down_->at(i) = NULL; } delete delete_these_arrays_on_tear_down_; delete_these_arrays_on_tear_down_ = NULL; } extensions_cache_.Initialize(false); // Yes, symmetrical } class Genesis BASE_EMBEDDED { public: Genesis(Handle global_object, v8::Handle global_template, v8::ExtensionConfiguration* extensions); ~Genesis() { } Handle result() { return result_; } Genesis* previous() { return previous_; } private: Handle global_context_; // There may be more than one active genesis object: When GC is // triggered during environment creation there may be weak handle // processing callbacks which may create new environments. Genesis* previous_; Handle global_context() { return global_context_; } // Creates some basic objects. Used for creating a context from scratch. void CreateRoots(); // Creates the empty function. Used for creating a context from scratch. Handle CreateEmptyFunction(); // Creates the ThrowTypeError function. ECMA 5th Ed. 13.2.3 Handle CreateThrowTypeErrorFunction(Builtins::Name builtin); void CreateStrictModeFunctionMaps(Handle empty); // Creates the global objects using the global and the template passed in // through the API. We call this regardless of whether we are building a // context from scratch or using a deserialized one from the partial snapshot // but in the latter case we don't use the objects it produces directly, as // we have to used the deserialized ones that are linked together with the // rest of the context snapshot. Handle CreateNewGlobals( v8::Handle global_template, Handle global_object, Handle* global_proxy_out); // Hooks the given global proxy into the context. If the context was created // by deserialization then this will unhook the global proxy that was // deserialized, leaving the GC to pick it up. void HookUpGlobalProxy(Handle inner_global, Handle global_proxy); // Similarly, we want to use the inner global that has been created by the // templates passed through the API. The inner global from the snapshot is // detached from the other objects in the snapshot. void HookUpInnerGlobal(Handle inner_global); // New context initialization. Used for creating a context from scratch. void InitializeGlobal(Handle inner_global, Handle empty_function); // Installs the contents of the native .js files on the global objects. // Used for creating a context from scratch. void InstallNativeFunctions(); bool InstallNatives(); void InstallBuiltinFunctionIds(); void InstallJSFunctionResultCaches(); void InitializeNormalizedMapCaches(); // Used both for deserialized and from-scratch contexts to add the extensions // provided. static bool InstallExtensions(Handle global_context, v8::ExtensionConfiguration* extensions); static bool InstallExtension(const char* name); static bool InstallExtension(v8::RegisteredExtension* current); static void InstallSpecialObjects(Handle global_context); bool InstallJSBuiltins(Handle builtins); bool ConfigureApiObject(Handle object, Handle object_template); bool ConfigureGlobalObjects(v8::Handle global_template); // Migrates all properties from the 'from' object to the 'to' // object and overrides the prototype in 'to' with the one from // 'from'. void TransferObject(Handle from, Handle to); void TransferNamedProperties(Handle from, Handle to); void TransferIndexedProperties(Handle from, Handle to); enum PrototypePropertyMode { DONT_ADD_PROTOTYPE, ADD_READONLY_PROTOTYPE, ADD_WRITEABLE_PROTOTYPE }; Handle CreateFunctionMap(PrototypePropertyMode prototype_mode); Handle ComputeFunctionInstanceDescriptor( PrototypePropertyMode prototypeMode); void MakeFunctionInstancePrototypeWritable(); Handle CreateStrictModeFunctionMap( PrototypePropertyMode prototype_mode, Handle empty_function, Handle arguments_callbacks, Handle caller_callbacks); Handle ComputeStrictFunctionInstanceDescriptor( PrototypePropertyMode propertyMode, Handle arguments, Handle caller); static bool CompileBuiltin(int index); static bool CompileNative(Vector name, Handle source); static bool CompileScriptCached(Vector name, Handle source, SourceCodeCache* cache, v8::Extension* extension, Handle top_context, bool use_runtime_context); Handle result_; // Function instance maps. Function literal maps are created initially with // a read only prototype for the processing of JS builtins. Later the function // instance maps are replaced in order to make prototype writable. // These are the final, writable prototype, maps. Handle function_instance_map_writable_prototype_; Handle strict_mode_function_instance_map_writable_prototype_; BootstrapperActive active_; friend class Bootstrapper; }; void Bootstrapper::Iterate(ObjectVisitor* v) { extensions_cache_.Iterate(v); v->Synchronize("Extensions"); } Handle Bootstrapper::CreateEnvironment( Handle global_object, v8::Handle global_template, v8::ExtensionConfiguration* extensions) { HandleScope scope; Handle env; Genesis genesis(global_object, global_template, extensions); env = genesis.result(); if (!env.is_null()) { if (InstallExtensions(env, extensions)) { return env; } } return Handle(); } static void SetObjectPrototype(Handle object, Handle proto) { // object.__proto__ = proto; Handle old_to_map = Handle(object->map()); Handle new_to_map = FACTORY->CopyMapDropTransitions(old_to_map); new_to_map->set_prototype(*proto); object->set_map(*new_to_map); } void Bootstrapper::DetachGlobal(Handle env) { Factory* factory = Isolate::Current()->factory(); JSGlobalProxy::cast(env->global_proxy())->set_context(*factory->null_value()); SetObjectPrototype(Handle(env->global_proxy()), factory->null_value()); env->set_global_proxy(env->global()); env->global()->set_global_receiver(env->global()); } void Bootstrapper::ReattachGlobal(Handle env, Handle global_object) { ASSERT(global_object->IsJSGlobalProxy()); Handle global = Handle::cast(global_object); env->global()->set_global_receiver(*global); env->set_global_proxy(*global); SetObjectPrototype(global, Handle(env->global())); global->set_context(*env); } static Handle InstallFunction(Handle target, const char* name, InstanceType type, int instance_size, Handle prototype, Builtins::Name call, bool is_ecma_native) { Isolate* isolate = Isolate::Current(); Factory* factory = isolate->factory(); Handle symbol = factory->LookupAsciiSymbol(name); Handle call_code = Handle(isolate->builtins()->builtin(call)); Handle function = prototype.is_null() ? factory->NewFunctionWithoutPrototype(symbol, call_code) : factory->NewFunctionWithPrototype(symbol, type, instance_size, prototype, call_code, is_ecma_native); SetLocalPropertyNoThrow(target, symbol, function, DONT_ENUM); if (is_ecma_native) { function->shared()->set_instance_class_name(*symbol); } return function; } Handle Genesis::ComputeFunctionInstanceDescriptor( PrototypePropertyMode prototypeMode) { Factory* factory = Isolate::Current()->factory(); Handle descriptors = factory->NewDescriptorArray(prototypeMode == DONT_ADD_PROTOTYPE ? 4 : 5); PropertyAttributes attributes = static_cast(DONT_ENUM | DONT_DELETE | READ_ONLY); { // Add length. Handle proxy = factory->NewProxy(&Accessors::FunctionLength); CallbacksDescriptor d(*factory->length_symbol(), *proxy, attributes); descriptors->Set(0, &d); } { // Add name. Handle proxy = factory->NewProxy(&Accessors::FunctionName); CallbacksDescriptor d(*factory->name_symbol(), *proxy, attributes); descriptors->Set(1, &d); } { // Add arguments. Handle proxy = factory->NewProxy(&Accessors::FunctionArguments); CallbacksDescriptor d(*factory->arguments_symbol(), *proxy, attributes); descriptors->Set(2, &d); } { // Add caller. Handle proxy = factory->NewProxy(&Accessors::FunctionCaller); CallbacksDescriptor d(*factory->caller_symbol(), *proxy, attributes); descriptors->Set(3, &d); } if (prototypeMode != DONT_ADD_PROTOTYPE) { // Add prototype. if (prototypeMode == ADD_WRITEABLE_PROTOTYPE) { attributes = static_cast(attributes & ~READ_ONLY); } Handle proxy = factory->NewProxy(&Accessors::FunctionPrototype); CallbacksDescriptor d(*factory->prototype_symbol(), *proxy, attributes); descriptors->Set(4, &d); } descriptors->Sort(); return descriptors; } Handle Genesis::CreateFunctionMap(PrototypePropertyMode prototype_mode) { Handle map = FACTORY->NewMap(JS_FUNCTION_TYPE, JSFunction::kSize); Handle descriptors = ComputeFunctionInstanceDescriptor(prototype_mode); map->set_instance_descriptors(*descriptors); map->set_function_with_prototype(prototype_mode != DONT_ADD_PROTOTYPE); return map; } Handle Genesis::CreateEmptyFunction() { // Allocate the map for function instances. Maps are allocated first and their // prototypes patched later, once empty function is created. // Please note that the prototype property for function instances must be // writable. Handle function_instance_map = CreateFunctionMap(ADD_WRITEABLE_PROTOTYPE); global_context()->set_function_instance_map(*function_instance_map); // Functions with this map will not have a 'prototype' property, and // can not be used as constructors. Handle function_without_prototype_map = CreateFunctionMap(DONT_ADD_PROTOTYPE); global_context()->set_function_without_prototype_map( *function_without_prototype_map); // Allocate the function map. This map is temporary, used only for processing // of builtins. // Later the map is replaced with writable prototype map, allocated below. Handle function_map = CreateFunctionMap(ADD_READONLY_PROTOTYPE); global_context()->set_function_map(*function_map); // The final map for functions. Writeable prototype. // This map is installed in MakeFunctionInstancePrototypeWritable. function_instance_map_writable_prototype_ = CreateFunctionMap(ADD_WRITEABLE_PROTOTYPE); Isolate* isolate = Isolate::Current(); Factory* factory = isolate->factory(); Heap* heap = isolate->heap(); Handle object_name = Handle(heap->Object_symbol()); { // --- O b j e c t --- Handle object_fun = factory->NewFunction(object_name, factory->null_value()); Handle object_function_map = factory->NewMap(JS_OBJECT_TYPE, JSObject::kHeaderSize); object_fun->set_initial_map(*object_function_map); object_function_map->set_constructor(*object_fun); global_context()->set_object_function(*object_fun); // Allocate a new prototype for the object function. Handle prototype = factory->NewJSObject( isolate->object_function(), TENURED); global_context()->set_initial_object_prototype(*prototype); SetPrototype(object_fun, prototype); object_function_map-> set_instance_descriptors(heap->empty_descriptor_array()); } // Allocate the empty function as the prototype for function ECMAScript // 262 15.3.4. Handle symbol = factory->LookupAsciiSymbol("Empty"); Handle empty_function = factory->NewFunctionWithoutPrototype(symbol, kNonStrictMode); // --- E m p t y --- Handle code = Handle(isolate->builtins()->builtin( Builtins::kEmptyFunction)); empty_function->set_code(*code); empty_function->shared()->set_code(*code); Handle source = factory->NewStringFromAscii(CStrVector("() {}")); Handle