Chromium Code Reviews| OLD | NEW |
|---|---|
| 1 // Copyright 2015 the V8 project authors. All rights reserved. | 1 // Copyright 2015 the V8 project authors. All rights reserved. |
| 2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
| 3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
| 4 | 4 |
| 5 #include "src/api-natives.h" | 5 #include "src/api-natives.h" |
| 6 #include "src/api.h" | 6 #include "src/api.h" |
| 7 #include "src/asmjs/asm-js.h" | 7 #include "src/asmjs/asm-js.h" |
| 8 #include "src/asmjs/asm-typer.h" | 8 #include "src/asmjs/asm-typer.h" |
| 9 #include "src/asmjs/asm-wasm-builder.h" | 9 #include "src/asmjs/asm-wasm-builder.h" |
| 10 #include "src/assert-scope.h" | 10 #include "src/assert-scope.h" |
| (...skipping 14 matching lines...) Expand all Loading... | |
| 25 #include "src/wasm/wasm-result.h" | 25 #include "src/wasm/wasm-result.h" |
| 26 | 26 |
| 27 typedef uint8_t byte; | 27 typedef uint8_t byte; |
| 28 | 28 |
| 29 using v8::internal::wasm::ErrorThrower; | 29 using v8::internal::wasm::ErrorThrower; |
| 30 | 30 |
| 31 namespace v8 { | 31 namespace v8 { |
| 32 | 32 |
| 33 namespace { | 33 namespace { |
| 34 | 34 |
| 35 #define ASSIGN(type, var, expr) \ | |
| 36 Local<type> var; \ | |
| 37 do { \ | |
| 38 if (!expr.ToLocal(&var)) return; \ | |
| 39 } while (false) | |
| 40 | |
| 41 #define DO_BOOL(expr) \ | |
| 42 do { \ | |
| 43 bool ok; \ | |
| 44 if (!expr.To(&ok) || !ok) return; \ | |
| 45 } while (false) | |
| 46 | |
| 35 // TODO(wasm): move brand check to the respective types, and don't throw | 47 // TODO(wasm): move brand check to the respective types, and don't throw |
| 36 // in it, rather, use a provided ErrorThrower, or let caller handle it. | 48 // in it, rather, use a provided ErrorThrower, or let caller handle it. |
| 37 static bool HasBrand(i::Handle<i::Object> value, i::Handle<i::Symbol> sym) { | 49 static bool HasBrand(i::Handle<i::Object> value, i::Handle<i::Symbol> sym) { |
| 38 if (!value->IsJSObject()) return false; | 50 if (!value->IsJSObject()) return false; |
| 39 i::Handle<i::JSObject> object = i::Handle<i::JSObject>::cast(value); | 51 i::Handle<i::JSObject> object = i::Handle<i::JSObject>::cast(value); |
| 40 Maybe<bool> has_brand = i::JSObject::HasOwnProperty(object, sym); | 52 Maybe<bool> has_brand = i::JSObject::HasOwnProperty(object, sym); |
| 41 return has_brand.FromMaybe(false); | 53 return has_brand.FromMaybe(false); |
| 42 } | 54 } |
| 43 | 55 |
| 44 static bool BrandCheck(i::Handle<i::Object> value, i::Handle<i::Symbol> sym, | 56 static bool BrandCheck(i::Handle<i::Object> value, i::Handle<i::Symbol> sym, |
| (...skipping 65 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 110 } | 122 } |
| 111 if (length > i::wasm::kV8MaxWasmModuleSize) { | 123 if (length > i::wasm::kV8MaxWasmModuleSize) { |
| 112 thrower->RangeError("buffer source exceeds maximum size of %zu (is %zu)", | 124 thrower->RangeError("buffer source exceeds maximum size of %zu (is %zu)", |
| 113 i::wasm::kV8MaxWasmModuleSize, length); | 125 i::wasm::kV8MaxWasmModuleSize, length); |
| 114 } | 126 } |
| 115 if (thrower->error()) return i::wasm::ModuleWireBytes(nullptr, nullptr); | 127 if (thrower->error()) return i::wasm::ModuleWireBytes(nullptr, nullptr); |
| 116 // TODO(titzer): use the handle as well? | 128 // TODO(titzer): use the handle as well? |
| 117 return i::wasm::ModuleWireBytes(start, start + length); | 129 return i::wasm::ModuleWireBytes(start, start + length); |
| 118 } | 130 } |
| 119 | 131 |
| 120 i::MaybeHandle<i::JSReceiver> GetSecondArgumentAsImports( | 132 i::MaybeHandle<i::JSReceiver> GetValueAsImports(const Local<Value>& arg, |
| 121 const v8::FunctionCallbackInfo<v8::Value>& args, ErrorThrower* thrower) { | 133 ErrorThrower* thrower) { |
| 122 if (args.Length() < 2) return {}; | 134 if (arg->IsUndefined()) return {}; |
| 123 if (args[1]->IsUndefined()) return {}; | |
| 124 | 135 |
| 125 if (!args[1]->IsObject()) { | 136 if (!arg->IsObject()) { |
| 126 thrower->TypeError("Argument 1 must be an object"); | 137 thrower->TypeError("Argument 1 must be an object"); |
| 127 return {}; | 138 return {}; |
| 128 } | 139 } |
| 129 Local<Object> obj = Local<Object>::Cast(args[1]); | 140 Local<Object> obj = Local<Object>::Cast(arg); |
| 130 return i::Handle<i::JSReceiver>::cast(v8::Utils::OpenHandle(*obj)); | 141 return i::Handle<i::JSReceiver>::cast(v8::Utils::OpenHandle(*obj)); |
| 131 } | 142 } |
| 132 | 143 |
| 133 // WebAssembly.compile(bytes) -> Promise | 144 // WebAssembly.compile(bytes) -> Promise |
| 134 void WebAssemblyCompile(const v8::FunctionCallbackInfo<v8::Value>& args) { | 145 void WebAssemblyCompile(const v8::FunctionCallbackInfo<v8::Value>& args) { |
| 135 v8::Isolate* isolate = args.GetIsolate(); | 146 v8::Isolate* isolate = args.GetIsolate(); |
| 136 i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate); | 147 i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate); |
| 137 if (i_isolate->wasm_compile_callback()(args)) return; | 148 if (i_isolate->wasm_compile_callback()(args)) return; |
| 138 | 149 |
| 139 HandleScope scope(isolate); | 150 HandleScope scope(isolate); |
| 140 ErrorThrower thrower(i_isolate, "WebAssembly.compile()"); | 151 ErrorThrower thrower(i_isolate, "WebAssembly.compile()"); |
| 141 | 152 |
| 142 Local<Context> context = isolate->GetCurrentContext(); | 153 Local<Context> context = isolate->GetCurrentContext(); |
| 143 v8::Local<v8::Promise::Resolver> resolver; | 154 ASSIGN(Promise::Resolver, resolver, Promise::Resolver::New(context)); |
| 144 if (!v8::Promise::Resolver::New(context).ToLocal(&resolver)) return; | |
| 145 v8::ReturnValue<v8::Value> return_value = args.GetReturnValue(); | 155 v8::ReturnValue<v8::Value> return_value = args.GetReturnValue(); |
| 146 return_value.Set(resolver->GetPromise()); | 156 return_value.Set(resolver->GetPromise()); |
| 147 | 157 |
| 148 auto bytes = GetFirstArgumentAsBytes(args, &thrower); | 158 auto bytes = GetFirstArgumentAsBytes(args, &thrower); |
| 149 if (thrower.error()) { | 159 if (thrower.error()) { |
| 150 auto maybe = resolver->Reject(context, Utils::ToLocal(thrower.Reify())); | 160 auto maybe = resolver->Reject(context, Utils::ToLocal(thrower.Reify())); |
| 151 CHECK_IMPLIES(!maybe.FromMaybe(false), | 161 CHECK_IMPLIES(!maybe.FromMaybe(false), |
| 152 i_isolate->has_scheduled_exception()); | 162 i_isolate->has_scheduled_exception()); |
| 153 return; | 163 return; |
| 154 } | 164 } |
| (...skipping 91 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 246 return; | 256 return; |
| 247 } | 257 } |
| 248 | 258 |
| 249 auto custom_sections = | 259 auto custom_sections = |
| 250 i::wasm::GetCustomSections(i_isolate, maybe_module.ToHandleChecked(), | 260 i::wasm::GetCustomSections(i_isolate, maybe_module.ToHandleChecked(), |
| 251 i::Handle<i::String>::cast(name), &thrower); | 261 i::Handle<i::String>::cast(name), &thrower); |
| 252 if (thrower.error()) return; | 262 if (thrower.error()) return; |
| 253 args.GetReturnValue().Set(Utils::ToLocal(custom_sections)); | 263 args.GetReturnValue().Set(Utils::ToLocal(custom_sections)); |
| 254 } | 264 } |
| 255 | 265 |
| 266 // Entered as internal implementation detail of sync and async instantiate. | |
| 267 // args[0] *must* be a WebAssembly.Module. | |
| 268 void WebAssemblyInstantiateImpl( | |
| 269 const v8::FunctionCallbackInfo<v8::Value>& args) { | |
| 270 DCHECK_GE(args.Length(), 1); | |
| 271 Local<Value> module = args[0]; | |
| 272 Local<Value> ffi = args.Data(); | |
| 273 | |
| 274 HandleScope scope(args.GetIsolate()); | |
| 275 v8::Isolate* isolate = args.GetIsolate(); | |
| 276 i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate); | |
| 277 ErrorThrower thrower(i_isolate, "WebAssembly Instantiation"); | |
| 278 i::MaybeHandle<i::JSReceiver> maybe_imports = | |
| 279 GetValueAsImports(ffi, &thrower); | |
| 280 if (thrower.error()) return; | |
| 281 | |
| 282 i::Handle<i::WasmModuleObject> module_obj = | |
| 283 i::Handle<i::WasmModuleObject>::cast( | |
| 284 Utils::OpenHandle(Object::Cast(*module))); | |
| 285 i::MaybeHandle<i::Object> instance_object = | |
| 286 i::wasm::SyncInstantiate(i_isolate, &thrower, module_obj, maybe_imports, | |
| 287 i::MaybeHandle<i::JSArrayBuffer>()); | |
| 288 | |
| 289 if (instance_object.is_null()) { | |
| 290 // TODO(wasm): this *should* mean there's an error to throw, but | |
| 291 // we exit sometimes the instantiation pipeline without throwing. | |
| 292 // v8:6232. | |
| 293 return; | |
| 294 } | |
| 295 args.GetReturnValue().Set(Utils::ToLocal(instance_object.ToHandleChecked())); | |
| 296 } | |
| 297 | |
| 298 void WebAssemblyInstantiateToPair( | |
| 299 const v8::FunctionCallbackInfo<v8::Value>& args) { | |
| 300 DCHECK_GE(args.Length(), 1); | |
| 301 Local<Value> module = args[0]; | |
| 302 Isolate* isolate = args.GetIsolate(); | |
| 303 Local<Context> context = isolate->GetCurrentContext(); | |
| 304 | |
| 305 const uint8_t* instance_str = reinterpret_cast<const uint8_t*>("instance"); | |
| 306 const uint8_t* module_str = reinterpret_cast<const uint8_t*>("module"); | |
| 307 ASSIGN(Function, vanilla_instantiate, | |
| 308 Function::New(context, WebAssemblyInstantiateImpl, args.Data())); | |
| 309 | |
| 310 ASSIGN(Value, instance, | |
| 311 vanilla_instantiate->Call(context, args.Holder(), 1, &module)); | |
| 312 Local<Object> ret = Object::New(isolate); | |
| 313 ASSIGN(String, instance_name, | |
| 314 String::NewFromOneByte(isolate, instance_str, | |
| 315 NewStringType::kInternalized)); | |
| 316 ASSIGN(String, module_name, | |
| 317 String::NewFromOneByte(isolate, module_str, | |
| 318 NewStringType::kInternalized)); | |
| 319 | |
| 320 DO_BOOL(ret->CreateDataProperty(context, instance_name, instance)); | |
| 321 DO_BOOL(ret->CreateDataProperty(context, module_name, module)); | |
| 322 args.GetReturnValue().Set(ret); | |
| 323 } | |
| 324 | |
| 256 // new WebAssembly.Instance(module, imports) -> WebAssembly.Instance | 325 // new WebAssembly.Instance(module, imports) -> WebAssembly.Instance |
| 257 void WebAssemblyInstance(const v8::FunctionCallbackInfo<v8::Value>& args) { | 326 void WebAssemblyInstance(const v8::FunctionCallbackInfo<v8::Value>& args) { |
| 258 HandleScope scope(args.GetIsolate()); | 327 HandleScope scope(args.GetIsolate()); |
| 259 v8::Isolate* isolate = args.GetIsolate(); | 328 Isolate* isolate = args.GetIsolate(); |
| 329 Local<Context> context = isolate->GetCurrentContext(); | |
| 260 i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate); | 330 i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate); |
| 261 if (i_isolate->wasm_instance_callback()(args)) return; | 331 if (i_isolate->wasm_instance_callback()(args)) return; |
| 262 | 332 |
| 263 ErrorThrower thrower(i_isolate, "WebAssembly.Instance()"); | 333 ErrorThrower thrower(i_isolate, "WebAssembly.Instance()"); |
| 264 | 334 |
| 265 auto maybe_module = GetFirstArgumentAsModule(args, &thrower); | 335 auto maybe_module = GetFirstArgumentAsModule(args, &thrower); |
| 266 if (thrower.error()) return; | 336 if (thrower.error()) return; |
| 267 | 337 |
| 268 auto maybe_imports = GetSecondArgumentAsImports(args, &thrower); | 338 // If args.Length < 2, this will be undefined - see FunctionCallbackInfo. |
| 269 if (thrower.error()) return; | 339 // We'll check for that in WebAssemblyInstantiateImpl. |
| 340 Local<Value> data = args[1]; | |
|
bradnelson
2017/04/11 23:18:41
You sure args does that automatically?
Old code di
Mircea Trofin
2017/04/11 23:33:18
Yes: https://cs.chromium.org/chromium/src/v8/inclu
| |
| 270 | 341 |
| 271 i::MaybeHandle<i::Object> instance_object = i::wasm::SyncInstantiate( | 342 ASSIGN(Function, impl, |
| 272 i_isolate, &thrower, maybe_module.ToHandleChecked(), maybe_imports, | 343 Function::New(context, WebAssemblyInstantiateImpl, data)); |
| 273 i::MaybeHandle<i::JSArrayBuffer>()); | 344 Local<Value> first_param = args[0]; |
| 274 if (instance_object.is_null()) return; | 345 ASSIGN(Value, ret, impl->Call(context, args.Holder(), 1, &first_param)); |
| 275 args.GetReturnValue().Set(Utils::ToLocal(instance_object.ToHandleChecked())); | 346 args.GetReturnValue().Set(ret); |
| 276 } | 347 } |
| 277 | 348 |
| 278 // WebAssembly.instantiate(module, imports) -> WebAssembly.Instance | 349 // WebAssembly.instantiate(module, imports) -> WebAssembly.Instance |
| 279 // WebAssembly.instantiate(bytes, imports) -> | 350 // WebAssembly.instantiate(bytes, imports) -> |
| 280 // {module: WebAssembly.Module, instance: WebAssembly.Instance} | 351 // {module: WebAssembly.Module, instance: WebAssembly.Instance} |
| 281 void WebAssemblyInstantiate(const v8::FunctionCallbackInfo<v8::Value>& args) { | 352 void WebAssemblyInstantiate(const v8::FunctionCallbackInfo<v8::Value>& args) { |
| 282 v8::Isolate* isolate = args.GetIsolate(); | 353 v8::Isolate* isolate = args.GetIsolate(); |
| 283 i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate); | 354 i::Isolate* i_isolate = reinterpret_cast<i::Isolate*>(isolate); |
| 284 if (i_isolate->wasm_instantiate_callback()(args)) return; | 355 if (i_isolate->wasm_instantiate_callback()(args)) return; |
| 285 | 356 |
| 286 ErrorThrower thrower(i_isolate, "WebAssembly.instantiate()"); | 357 ErrorThrower thrower(i_isolate, "WebAssembly.instantiate()"); |
| 287 | 358 |
| 288 HandleScope scope(isolate); | 359 HandleScope scope(isolate); |
| 289 | 360 |
| 290 Local<Context> context = isolate->GetCurrentContext(); | 361 Local<Context> context = isolate->GetCurrentContext(); |
| 291 i::Handle<i::Context> i_context = Utils::OpenHandle(*context); | 362 i::Handle<i::Context> i_context = Utils::OpenHandle(*context); |
| 292 | 363 |
| 293 v8::Local<v8::Promise::Resolver> resolver; | 364 ASSIGN(Promise::Resolver, resolver, Promise::Resolver::New(context)); |
| 294 if (!v8::Promise::Resolver::New(context).ToLocal(&resolver)) return; | 365 Local<Promise> module_promise = resolver->GetPromise(); |
| 295 v8::ReturnValue<v8::Value> return_value = args.GetReturnValue(); | 366 args.GetReturnValue().Set(module_promise); |
| 296 return_value.Set(resolver->GetPromise()); | |
| 297 | 367 |
| 298 if (args.Length() < 1) { | 368 if (args.Length() < 1) { |
| 299 thrower.TypeError( | 369 thrower.TypeError( |
| 300 "Argument 0 must be provided and must be either a buffer source or a " | 370 "Argument 0 must be provided and must be either a buffer source or a " |
| 301 "WebAssembly.Module object"); | 371 "WebAssembly.Module object"); |
| 302 auto maybe = resolver->Reject(context, Utils::ToLocal(thrower.Reify())); | 372 auto maybe = resolver->Reject(context, Utils::ToLocal(thrower.Reify())); |
| 303 CHECK_IMPLIES(!maybe.FromMaybe(false), | 373 CHECK_IMPLIES(!maybe.FromMaybe(false), |
| 304 i_isolate->has_scheduled_exception()); | 374 i_isolate->has_scheduled_exception()); |
| 305 return; | 375 return; |
| 306 } | 376 } |
| 307 | 377 |
| 308 i::Handle<i::Object> first_arg = Utils::OpenHandle(*args[0]); | 378 Local<Value> first_arg_value = args[0]; |
| 379 i::Handle<i::Object> first_arg = Utils::OpenHandle(*first_arg_value); | |
| 309 if (!first_arg->IsJSObject()) { | 380 if (!first_arg->IsJSObject()) { |
| 310 thrower.TypeError( | 381 thrower.TypeError( |
| 311 "Argument 0 must be a buffer source or a WebAssembly.Module object"); | 382 "Argument 0 must be a buffer source or a WebAssembly.Module object"); |
| 312 auto maybe = resolver->Reject(context, Utils::ToLocal(thrower.Reify())); | 383 auto maybe = resolver->Reject(context, Utils::ToLocal(thrower.Reify())); |
| 313 CHECK_IMPLIES(!maybe.FromMaybe(false), | 384 CHECK_IMPLIES(!maybe.FromMaybe(false), |
| 314 i_isolate->has_scheduled_exception()); | 385 i_isolate->has_scheduled_exception()); |
| 315 return; | 386 return; |
| 316 } | 387 } |
| 317 | 388 |
| 318 auto maybe_imports = GetSecondArgumentAsImports(args, &thrower); | 389 FunctionCallback instantiator = nullptr; |
| 319 if (thrower.error()) { | 390 if (HasBrand(first_arg, i::Handle<i::Symbol>(i_context->wasm_module_sym()))) { |
| 320 auto maybe = resolver->Reject(context, Utils::ToLocal(thrower.Reify())); | 391 module_promise = resolver->GetPromise(); |
| 321 CHECK_IMPLIES(!maybe.FromMaybe(false), | 392 DO_BOOL(resolver->Resolve(context, first_arg_value)); |
| 322 i_isolate->has_scheduled_exception()); | 393 instantiator = WebAssemblyInstantiateImpl; |
| 323 return; | 394 } else { |
| 395 ASSIGN(Function, async_compile, Function::New(context, WebAssemblyCompile)); | |
| 396 ASSIGN(Value, async_compile_retval, | |
| 397 async_compile->Call(context, args.Holder(), 1, &first_arg_value)); | |
| 398 module_promise = Local<Promise>::Cast(async_compile_retval); | |
| 399 instantiator = WebAssemblyInstantiateToPair; | |
| 324 } | 400 } |
| 325 i::Handle<i::JSPromise> promise = Utils::OpenHandle(*resolver->GetPromise()); | 401 DCHECK(!module_promise.IsEmpty()); |
| 326 if (HasBrand(first_arg, i::Handle<i::Symbol>(i_context->wasm_module_sym()))) { | 402 DCHECK_NOT_NULL(instantiator); |
| 327 // WebAssembly.instantiate(module, imports) -> WebAssembly.Instance | 403 // If args.Length < 2, this will be undefined - see FunctionCallbackInfo. |
| 328 auto module_object = GetFirstArgumentAsModule(args, &thrower); | 404 // We'll check for that in WebAssemblyInstantiateImpl. |
| 329 i::wasm::AsyncInstantiate(i_isolate, promise, | 405 Local<Value> data = args[1]; |
|
bradnelson
2017/04/11 23:18:41
Same
Mircea Trofin
2017/04/11 23:33:18
See above.
| |
| 330 module_object.ToHandleChecked(), maybe_imports); | 406 ASSIGN(Function, instantiate_impl, |
| 331 } else { | 407 Function::New(context, instantiator, data)); |
| 332 // WebAssembly.instantiate(bytes, imports) -> {module, instance} | 408 ASSIGN(Promise, result, module_promise->Then(context, instantiate_impl)); |
| 333 auto bytes = GetFirstArgumentAsBytes(args, &thrower); | 409 args.GetReturnValue().Set(result); |
| 334 if (thrower.error()) { | |
| 335 auto maybe = resolver->Reject(context, Utils::ToLocal(thrower.Reify())); | |
| 336 CHECK_IMPLIES(!maybe.FromMaybe(false), | |
| 337 i_isolate->has_scheduled_exception()); | |
| 338 return; | |
| 339 } | |
| 340 i::wasm::AsyncCompileAndInstantiate(i_isolate, promise, bytes, | |
| 341 maybe_imports); | |
| 342 } | |
| 343 } | 410 } |
| 344 | 411 |
| 345 bool GetIntegerProperty(v8::Isolate* isolate, ErrorThrower* thrower, | 412 bool GetIntegerProperty(v8::Isolate* isolate, ErrorThrower* thrower, |
| 346 Local<Context> context, Local<v8::Object> object, | 413 Local<Context> context, Local<v8::Object> object, |
| 347 Local<String> property, int* result, | 414 Local<String> property, int* result, |
| 348 int64_t lower_bound, uint64_t upper_bound) { | 415 int64_t lower_bound, uint64_t upper_bound) { |
| 349 v8::MaybeLocal<v8::Value> maybe = object->Get(context, property); | 416 v8::MaybeLocal<v8::Value> maybe = object->Get(context, property); |
| 350 v8::Local<v8::Value> value; | 417 v8::Local<v8::Value> value; |
| 351 if (maybe.ToLocal(&value)) { | 418 if (maybe.ToLocal(&value)) { |
| 352 int64_t number; | 419 int64_t number; |
| (...skipping 531 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... | |
| 884 i::Handle<i::Symbol> symbol(isolate->context()->wasm_memory_sym(), isolate); | 951 i::Handle<i::Symbol> symbol(isolate->context()->wasm_memory_sym(), isolate); |
| 885 return HasBrand(value, symbol); | 952 return HasBrand(value, symbol); |
| 886 } | 953 } |
| 887 | 954 |
| 888 bool WasmJs::IsWasmTableObject(Isolate* isolate, Handle<Object> value) { | 955 bool WasmJs::IsWasmTableObject(Isolate* isolate, Handle<Object> value) { |
| 889 i::Handle<i::Symbol> symbol(isolate->context()->wasm_table_sym(), isolate); | 956 i::Handle<i::Symbol> symbol(isolate->context()->wasm_table_sym(), isolate); |
| 890 return HasBrand(value, symbol); | 957 return HasBrand(value, symbol); |
| 891 } | 958 } |
| 892 } // namespace internal | 959 } // namespace internal |
| 893 } // namespace v8 | 960 } // namespace v8 |
| OLD | NEW |