 Chromium Code Reviews
 Chromium Code Reviews Issue 2084573002:
  Upgrade Wasm JS API, step 1  (Closed) 
  Base URL: https://chromium.googlesource.com/v8/v8.git@master
    
  
    Issue 2084573002:
  Upgrade Wasm JS API, step 1  (Closed) 
  Base URL: https://chromium.googlesource.com/v8/v8.git@master| Index: src/wasm/wasm-js.cc | 
| diff --git a/src/wasm/wasm-js.cc b/src/wasm/wasm-js.cc | 
| index 2c83f823556c852a38660ca87b18341ab87fb5db..6dc14952a01e943b8bd5a9fc012c8d648c59dbae 100644 | 
| --- a/src/wasm/wasm-js.cc | 
| +++ b/src/wasm/wasm-js.cc | 
| @@ -35,30 +35,25 @@ struct RawBuffer { | 
| size_t size() { return static_cast<size_t>(end - start); } | 
| }; | 
| -RawBuffer GetRawBufferArgument( | 
| - ErrorThrower& thrower, const v8::FunctionCallbackInfo<v8::Value>& args) { | 
| - if (args.Length() < 1) { | 
| - thrower.Error("Argument 0 must be an array buffer"); | 
| - return {nullptr, nullptr}; | 
| - } | 
| - | 
| +RawBuffer GetRawBufferSource( | 
| + v8::Local<v8::Value> source, ErrorThrower* thrower) { | 
| const byte* start = nullptr; | 
| const byte* end = nullptr; | 
| - if (args[0]->IsArrayBuffer()) { | 
| + if (source->IsArrayBuffer()) { | 
| // A raw array buffer was passed. | 
| - Local<ArrayBuffer> buffer = Local<ArrayBuffer>::Cast(args[0]); | 
| + Local<ArrayBuffer> buffer = Local<ArrayBuffer>::Cast(source); | 
| ArrayBuffer::Contents contents = buffer->GetContents(); | 
| start = reinterpret_cast<const byte*>(contents.Data()); | 
| end = start + contents.ByteLength(); | 
| if (start == nullptr || end == start) { | 
| - thrower.Error("ArrayBuffer argument is empty"); | 
| + thrower->Error("ArrayBuffer argument is empty"); | 
| } | 
| - } else if (args[0]->IsTypedArray()) { | 
| + } else if (source->IsTypedArray()) { | 
| // A TypedArray was passed. | 
| - Local<TypedArray> array = Local<TypedArray>::Cast(args[0]); | 
| + Local<TypedArray> array = Local<TypedArray>::Cast(source); | 
| Local<ArrayBuffer> buffer = array->Buffer(); | 
| ArrayBuffer::Contents contents = buffer->GetContents(); | 
| @@ -68,10 +63,10 @@ RawBuffer GetRawBufferArgument( | 
| end = start + array->ByteLength(); | 
| if (start == nullptr || end == start) { | 
| - thrower.Error("ArrayBuffer argument is empty"); | 
| + thrower->Error("ArrayBuffer argument is empty"); | 
| } | 
| } else { | 
| - thrower.Error("Argument 0 must be an ArrayBuffer or Uint8Array"); | 
| + thrower->Error("Argument 0 must be an ArrayBuffer or Uint8Array"); | 
| } | 
| return {start, end}; | 
| @@ -80,9 +75,13 @@ RawBuffer GetRawBufferArgument( | 
| void VerifyModule(const v8::FunctionCallbackInfo<v8::Value>& args) { | 
| HandleScope scope(args.GetIsolate()); | 
| i::Isolate* isolate = reinterpret_cast<i::Isolate*>(args.GetIsolate()); | 
| - ErrorThrower thrower(isolate, "WASM.verifyModule()"); | 
| + ErrorThrower thrower(isolate, "Wasm.verifyModule()"); | 
| - RawBuffer buffer = GetRawBufferArgument(thrower, args); | 
| + if (args.Length() < 1) { | 
| + thrower.Error("Argument 0 must be a buffer source"); | 
| + return; | 
| + } | 
| + RawBuffer buffer = GetRawBufferSource(args[0], &thrower); | 
| if (thrower.error()) return; | 
| i::Zone zone(isolate->allocator()); | 
| @@ -100,9 +99,13 @@ void VerifyModule(const v8::FunctionCallbackInfo<v8::Value>& args) { | 
| void VerifyFunction(const v8::FunctionCallbackInfo<v8::Value>& args) { | 
| HandleScope scope(args.GetIsolate()); | 
| i::Isolate* isolate = reinterpret_cast<i::Isolate*>(args.GetIsolate()); | 
| - ErrorThrower thrower(isolate, "WASM.verifyFunction()"); | 
| + ErrorThrower thrower(isolate, "Wasm.verifyFunction()"); | 
| - RawBuffer buffer = GetRawBufferArgument(thrower, args); | 
| + if (args.Length() < 1) { | 
| + thrower.Error("Argument 0 must be a buffer source"); | 
| + return; | 
| + } | 
| + RawBuffer buffer = GetRawBufferSource(args[0], &thrower); | 
| if (thrower.error()) return; | 
| internal::wasm::FunctionResult result; | 
| @@ -160,16 +163,9 @@ v8::internal::wasm::ZoneBuffer* TranslateAsmModule( | 
| i::MaybeHandle<i::JSObject> InstantiateModuleCommon( | 
| const v8::FunctionCallbackInfo<v8::Value>& args, const byte* start, | 
| const byte* end, ErrorThrower* thrower, | 
| - internal::wasm::ModuleOrigin origin) { | 
| + internal::wasm::ModuleOrigin origin = i::wasm::kWasmOrigin) { | 
| i::Isolate* isolate = reinterpret_cast<i::Isolate*>(args.GetIsolate()); | 
| - i::Handle<i::JSArrayBuffer> memory = i::Handle<i::JSArrayBuffer>::null(); | 
| - if (args.Length() > 2 && args[2]->IsArrayBuffer()) { | 
| - Local<Object> obj = Local<Object>::Cast(args[2]); | 
| - i::Handle<i::Object> mem_obj = v8::Utils::OpenHandle(*obj); | 
| - memory = i::Handle<i::JSArrayBuffer>(i::JSArrayBuffer::cast(*mem_obj)); | 
| - } | 
| - | 
| // Decode but avoid a redundant pass over function bodies for verification. | 
| // Verification will happen during compilation. | 
| i::Zone zone(isolate->allocator()); | 
| @@ -189,8 +185,14 @@ i::MaybeHandle<i::JSObject> InstantiateModuleCommon( | 
| ffi = i::Handle<i::JSReceiver>::cast(v8::Utils::OpenHandle(*obj)); | 
| } | 
| - object = result.val->Instantiate(isolate, ffi, memory); | 
| + i::Handle<i::JSArrayBuffer> memory = i::Handle<i::JSArrayBuffer>::null(); | 
| + if (args.Length() > 2 && args[2]->IsArrayBuffer()) { | 
| + Local<Object> obj = Local<Object>::Cast(args[2]); | 
| + i::Handle<i::Object> mem_obj = v8::Utils::OpenHandle(*obj); | 
| + memory = i::Handle<i::JSArrayBuffer>(i::JSArrayBuffer::cast(*mem_obj)); | 
| + } | 
| + object = result.val->Instantiate(isolate, ffi, memory); | 
| if (!object.is_null()) { | 
| args.GetReturnValue().Set(v8::Utils::ToLocal(object.ToHandleChecked())); | 
| } | 
| @@ -203,7 +205,7 @@ i::MaybeHandle<i::JSObject> InstantiateModuleCommon( | 
| void InstantiateModuleFromAsm(const v8::FunctionCallbackInfo<v8::Value>& args) { | 
| HandleScope scope(args.GetIsolate()); | 
| i::Isolate* isolate = reinterpret_cast<i::Isolate*>(args.GetIsolate()); | 
| - ErrorThrower thrower(isolate, "WASM.instantiateModuleFromAsm()"); | 
| + ErrorThrower thrower(isolate, "Wasm.instantiateModuleFromAsm()"); | 
| if (!args[0]->IsString()) { | 
| thrower.Error("Asm module text should be a string"); | 
| @@ -275,13 +277,104 @@ void InstantiateModuleFromAsm(const v8::FunctionCallbackInfo<v8::Value>& args) { | 
| void InstantiateModule(const v8::FunctionCallbackInfo<v8::Value>& args) { | 
| HandleScope scope(args.GetIsolate()); | 
| i::Isolate* isolate = reinterpret_cast<i::Isolate*>(args.GetIsolate()); | 
| - ErrorThrower thrower(isolate, "WASM.instantiateModule()"); | 
| + ErrorThrower thrower(isolate, "Wasm.instantiateModule()"); | 
| + | 
| + if (args.Length() < 1) { | 
| + thrower.Error("Argument 0 must be a buffer source"); | 
| + return; | 
| + } | 
| + RawBuffer buffer = GetRawBufferSource(args[0], &thrower); | 
| + if (buffer.start == nullptr) return; | 
| + | 
| + InstantiateModuleCommon(args, buffer.start, buffer.end, &thrower); | 
| +} | 
| + | 
| + | 
| +static i::MaybeHandle<i::JSObject> CreateModuleObject( | 
| + v8::Isolate* isolate, const v8::Local<v8::Value> source, | 
| + ErrorThrower* thrower) { | 
| + i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate); | 
| + | 
| + RawBuffer buffer = GetRawBufferSource(source, thrower); | 
| + if (buffer.start == nullptr) return i::MaybeHandle<i::JSObject>(); | 
| + | 
| + // TODO(rossberg): Once we can, do compilation here. | 
| + DCHECK(source->IsArrayBuffer() || source->IsTypedArray()); | 
| + Local<Context> context = isolate->GetCurrentContext(); | 
| + i::Handle<i::Context> i_context = Utils::OpenHandle(*context); | 
| + i::Handle<i::JSFunction> module_cons(i_context->wasm_module_constructor()); | 
| + i::Handle<i::JSObject> module_obj = | 
| + i_isolate->factory()->NewJSObject(module_cons); | 
| + i::Handle<i::Object> module_ref = Utils::OpenHandle(*source); | 
| + i::Handle<i::Symbol> module_sym(i_context->wasm_module_sym()); | 
| + i::Object::SetProperty(module_obj, module_sym, module_ref, i::STRICT).Check(); | 
| + | 
| + return module_obj; | 
| +} | 
| + | 
| +void WebAssemblyCompile(const v8::FunctionCallbackInfo<v8::Value>& args) { | 
| + v8::Isolate* isolate = args.GetIsolate(); | 
| + HandleScope scope(isolate); | 
| + ErrorThrower thrower(reinterpret_cast<i::Isolate*>(isolate), | 
| + "WebAssembly.compile()"); | 
| + | 
| + if (args.Length() < 1) { | 
| + thrower.Error("Argument 0 must be a buffer source"); | 
| + return; | 
| + } | 
| + i::MaybeHandle<i::JSObject> module_obj = | 
| + CreateModuleObject(isolate, args[0], &thrower); | 
| + if (module_obj.is_null()) return; | 
| + | 
| + Local<Context> context = isolate->GetCurrentContext(); | 
| + v8::Local<v8::Promise::Resolver> resolver; | 
| + if (!v8::Promise::Resolver::New(context).ToLocal(&resolver)) return; | 
| + resolver->Resolve(context, Utils::ToLocal(module_obj.ToHandleChecked())); | 
| + | 
| + v8::ReturnValue<v8::Value> return_value = args.GetReturnValue(); | 
| + return_value.Set(resolver->GetPromise()); | 
| +} | 
| + | 
| +void WebAssemblyModule(const v8::FunctionCallbackInfo<v8::Value>& args) { | 
| + v8::Isolate* isolate = args.GetIsolate(); | 
| + HandleScope scope(isolate); | 
| + ErrorThrower thrower(reinterpret_cast<i::Isolate*>(isolate), | 
| + "WebAssembly.Module()"); | 
| + | 
| + if (args.Length() < 1) { | 
| + thrower.Error("Argument 0 must be a buffer source"); | 
| + return; | 
| + } | 
| + i::MaybeHandle<i::JSObject> module_obj = | 
| + CreateModuleObject(isolate, args[0], &thrower); | 
| + if (module_obj.is_null()) return; | 
| + | 
| + v8::ReturnValue<v8::Value> return_value = args.GetReturnValue(); | 
| + return_value.Set(Utils::ToLocal(module_obj.ToHandleChecked())); | 
| +} | 
| + | 
| +void WebAssemblyInstance(const v8::FunctionCallbackInfo<v8::Value>& args) { | 
| + HandleScope scope(args.GetIsolate()); | 
| + v8::Isolate* isolate = args.GetIsolate(); | 
| + ErrorThrower thrower(reinterpret_cast<i::Isolate*>(isolate), | 
| + "WebAssembly.Instance()"); | 
| - RawBuffer buffer = GetRawBufferArgument(thrower, args); | 
| + if (args.Length() < 1) { | 
| + thrower.Error("Argument 0 must be a WebAssembly.Module"); | 
| + return; | 
| + } | 
| + Local<Context> context = isolate->GetCurrentContext(); | 
| + i::Handle<i::Context> i_context = Utils::OpenHandle(*context); | 
| + i::Handle<i::Symbol> module_sym(i_context->wasm_module_sym()); | 
| + i::MaybeHandle<i::Object> source = | 
| + i::Object::GetProperty(Utils::OpenHandle(*args[0]), module_sym); | 
| + if (source.is_null()) return; | 
| + | 
| + RawBuffer buffer = | 
| + GetRawBufferSource(Utils::ToLocal(source.ToHandleChecked()), &thrower); | 
| if (buffer.start == nullptr) return; | 
| - InstantiateModuleCommon(args, buffer.start, buffer.end, &thrower, | 
| - internal::wasm::kWasmOrigin); | 
| + InstantiateModuleCommon(args, buffer.start, buffer.end, &thrower); | 
| } | 
| } // namespace | 
| @@ -299,8 +392,8 @@ static Handle<String> v8_str(Isolate* isolate, const char* str) { | 
| return isolate->factory()->NewStringFromAsciiChecked(str); | 
| } | 
| -static void InstallFunc(Isolate* isolate, Handle<JSObject> object, | 
| - const char* str, FunctionCallback func) { | 
| +static Handle<JSFunction> InstallFunc(Isolate* isolate, Handle<JSObject> object, | 
| + const char* str, FunctionCallback func) { | 
| Handle<String> name = v8_str(isolate, str); | 
| Handle<FunctionTemplateInfo> temp = NewTemplate(isolate, func); | 
| Handle<JSFunction> function = | 
| @@ -308,16 +401,54 @@ static void InstallFunc(Isolate* isolate, Handle<JSObject> object, | 
| PropertyAttributes attributes = | 
| static_cast<PropertyAttributes>(DONT_DELETE | READ_ONLY); | 
| JSObject::AddProperty(object, name, function, attributes); | 
| + return function; | 
| } | 
| void WasmJs::Install(Isolate* isolate, Handle<JSGlobalObject> global) { | 
| + Factory* factory = isolate->factory(); | 
| + | 
| // Setup wasm function map. | 
| Handle<Context> context(global->native_context(), isolate); | 
| InstallWasmFunctionMap(isolate, context); | 
| - // Bind the WASM object. | 
| - Factory* factory = isolate->factory(); | 
| - Handle<String> name = v8_str(isolate, "Wasm"); | 
| + // Bind the experimental WASM object. | 
| + // TODO(rossberg, titzer): remove once it's no longer needed. | 
| + { | 
| + Handle<String> name = v8_str(isolate, "Wasm"); | 
| + Handle<JSFunction> cons = factory->NewFunction(name); | 
| + JSFunction::SetInstancePrototype( | 
| + cons, Handle<Object>(context->initial_object_prototype(), isolate)); | 
| + cons->shared()->set_instance_class_name(*name); | 
| + Handle<JSObject> wasm_object = factory->NewJSObject(cons, TENURED); | 
| + PropertyAttributes attributes = static_cast<PropertyAttributes>(DONT_ENUM); | 
| + JSObject::AddProperty(global, name, wasm_object, attributes); | 
| + | 
| + // Install functions on the WASM object. | 
| + InstallFunc(isolate, wasm_object, "verifyModule", VerifyModule); | 
| + InstallFunc(isolate, wasm_object, "verifyFunction", VerifyFunction); | 
| + InstallFunc(isolate, wasm_object, "instantiateModule", InstantiateModule); | 
| + InstallFunc(isolate, wasm_object, "instantiateModuleFromAsm", | 
| + InstantiateModuleFromAsm); | 
| + | 
| + { | 
| + // Add the Wasm.experimentalVersion property. | 
| + Handle<String> name = v8_str(isolate, "experimentalVersion"); | 
| + PropertyAttributes attributes = | 
| + static_cast<PropertyAttributes>(DONT_DELETE | READ_ONLY); | 
| + Handle<Smi> value = | 
| + Handle<Smi>(Smi::FromInt(wasm::kWasmVersion), isolate); | 
| + JSObject::AddProperty(wasm_object, name, value, attributes); | 
| + } | 
| + } | 
| + | 
| + // Create private symbols. | 
| + Handle<Symbol> module_sym = isolate->factory()->NewPrivateSymbol(); | 
| + Handle<Symbol> instance_sym = isolate->factory()->NewPrivateSymbol(); | 
| + context->set_wasm_module_sym(*module_sym); | 
| + context->set_wasm_instance_sym(*instance_sym); | 
| + | 
| + // Bind the WebAssembly object. | 
| + Handle<String> name = v8_str(isolate, "WebAssembly"); | 
| Handle<JSFunction> cons = factory->NewFunction(name); | 
| JSFunction::SetInstancePrototype( | 
| cons, Handle<Object>(context->initial_object_prototype(), isolate)); | 
| @@ -326,21 +457,14 @@ void WasmJs::Install(Isolate* isolate, Handle<JSGlobalObject> global) { | 
| PropertyAttributes attributes = static_cast<PropertyAttributes>(DONT_ENUM); | 
| JSObject::AddProperty(global, name, wasm_object, attributes); | 
| - // Install functions on the WASM object. | 
| - InstallFunc(isolate, wasm_object, "verifyModule", VerifyModule); | 
| - InstallFunc(isolate, wasm_object, "verifyFunction", VerifyFunction); | 
| - InstallFunc(isolate, wasm_object, "instantiateModule", InstantiateModule); | 
| - InstallFunc(isolate, wasm_object, "instantiateModuleFromAsm", | 
| - InstantiateModuleFromAsm); | 
| - | 
| - { | 
| - // Add the Wasm.experimentalVersion property. | 
| - Handle<String> name = v8_str(isolate, "experimentalVersion"); | 
| - PropertyAttributes attributes = | 
| - static_cast<PropertyAttributes>(DONT_DELETE | READ_ONLY); | 
| - Handle<Smi> value = Handle<Smi>(Smi::FromInt(wasm::kWasmVersion), isolate); | 
| - JSObject::AddProperty(wasm_object, name, value, attributes); | 
| - } | 
| + // Install static methods on WebAssembly object. | 
| + InstallFunc(isolate, wasm_object, "compile", WebAssemblyCompile); | 
| + Handle<JSFunction> module_constructor = | 
| + InstallFunc(isolate, wasm_object, "Module", WebAssemblyModule); | 
| + Handle<JSFunction> instance_constructor = | 
| + InstallFunc(isolate, wasm_object, "Instance", WebAssemblyInstance); | 
| 
Mircea Trofin
2016/06/21 13:20:49
Don't we want to name this "Instantiate", rather t
 
rossberg
2016/06/21 13:35:16
No, it's a constructor, i.e., represents a class.
 | 
| + context->set_wasm_module_constructor(*module_constructor); | 
| + context->set_wasm_instance_constructor(*instance_constructor); | 
| } | 
| void WasmJs::InstallWasmFunctionMap(Isolate* isolate, Handle<Context> context) { |