From 642076f2aff3cddf93dd7cb6b1a332f8064ab39f Mon Sep 17 00:00:00 2001 From: Ben Noordhuis Date: Fri, 15 Apr 2016 19:40:01 +0200 Subject: [PATCH] src: don't set non-primitive values on templates V8 is going to disallow non-primitive values on v8::FunctionTemplate and v8::ObjectTemplate because those can't be shared across contexts. Fixes: https://github.com/nodejs/node/issues/6216 PR-URL: https://github.com/nodejs/node/pull/6228 Reviewed-By: Trevor Norris --- lib/_http_common.js | 10 ++++++---- src/env-inl.h | 14 ++++++-------- src/node.h | 21 ++++++++++++++++----- src/node_http_parser.cc | 2 +- test/parallel/test-http-parser.js | 6 +++--- 5 files changed, 32 insertions(+), 21 deletions(-) diff --git a/lib/_http_common.js b/lib/_http_common.js index 08f93d8c4d04a2..7c9fac5c6b7671 100644 --- a/lib/_http_common.js +++ b/lib/_http_common.js @@ -1,8 +1,10 @@ 'use strict'; -const FreeList = require('internal/freelist').FreeList; -const HTTPParser = process.binding('http_parser').HTTPParser; +const binding = process.binding('http_parser'); +const methods = binding.methods; +const HTTPParser = binding.HTTPParser; +const FreeList = require('internal/freelist').FreeList; const incoming = require('_http_incoming'); const IncomingMessage = incoming.IncomingMessage; const readStart = incoming.readStart; @@ -14,7 +16,7 @@ exports.debug = debug; exports.CRLF = '\r\n'; exports.chunkExpression = /chunk/i; exports.continueExpression = /100-continue/i; -exports.methods = HTTPParser.methods; +exports.methods = methods; const kOnHeaders = HTTPParser.kOnHeaders | 0; const kOnHeadersComplete = HTTPParser.kOnHeadersComplete | 0; @@ -71,7 +73,7 @@ function parserOnHeadersComplete(versionMajor, versionMinor, headers, method, if (typeof method === 'number') { // server only - parser.incoming.method = HTTPParser.methods[method]; + parser.incoming.method = methods[method]; } else { // client only parser.incoming.statusCode = statusCode; diff --git a/src/env-inl.h b/src/env-inl.h index 475e8e83f594e6..2c6248ae608306 100644 --- a/src/env-inl.h +++ b/src/env-inl.h @@ -507,27 +507,25 @@ inline void Environment::SetProtoMethod(v8::Local that, const char* name, v8::FunctionCallback callback) { v8::Local signature = v8::Signature::New(isolate(), that); - v8::Local function = - NewFunctionTemplate(callback, signature)->GetFunction(); + v8::Local t = NewFunctionTemplate(callback, signature); // kInternalized strings are created in the old space. const v8::NewStringType type = v8::NewStringType::kInternalized; v8::Local name_string = v8::String::NewFromUtf8(isolate(), name, type).ToLocalChecked(); - that->PrototypeTemplate()->Set(name_string, function); - function->SetName(name_string); // NODE_SET_PROTOTYPE_METHOD() compatibility. + that->PrototypeTemplate()->Set(name_string, t); + t->SetClassName(name_string); // NODE_SET_PROTOTYPE_METHOD() compatibility. } inline void Environment::SetTemplateMethod(v8::Local that, const char* name, v8::FunctionCallback callback) { - v8::Local function = - NewFunctionTemplate(callback)->GetFunction(); + v8::Local t = NewFunctionTemplate(callback); // kInternalized strings are created in the old space. const v8::NewStringType type = v8::NewStringType::kInternalized; v8::Local name_string = v8::String::NewFromUtf8(isolate(), name, type).ToLocalChecked(); - that->Set(name_string, function); - function->SetName(name_string); // NODE_SET_METHOD() compatibility. + that->Set(name_string, t); + t->SetClassName(name_string); // NODE_SET_METHOD() compatibility. } inline v8::Local Environment::NewInternalFieldObject() { diff --git a/src/node.h b/src/node.h index 59df8e18b4f626..529ee75f279053 100644 --- a/src/node.h +++ b/src/node.h @@ -240,8 +240,20 @@ NODE_EXTERN void RunAtExit(Environment* env); while (0) // Used to be a macro, hence the uppercase name. -template -inline void NODE_SET_METHOD(const TypeName& recv, +inline void NODE_SET_METHOD(v8::Local recv, + const char* name, + v8::FunctionCallback callback) { + v8::Isolate* isolate = v8::Isolate::GetCurrent(); + v8::HandleScope handle_scope(isolate); + v8::Local t = v8::FunctionTemplate::New(isolate, + callback); + v8::Local fn_name = v8::String::NewFromUtf8(isolate, name); + t->SetClassName(fn_name); + recv->Set(fn_name, t); +} + +// Used to be a macro, hence the uppercase name. +inline void NODE_SET_METHOD(v8::Local recv, const char* name, v8::FunctionCallback callback) { v8::Isolate* isolate = v8::Isolate::GetCurrent(); @@ -265,10 +277,9 @@ inline void NODE_SET_PROTOTYPE_METHOD(v8::Local recv, v8::Local s = v8::Signature::New(isolate, recv); v8::Local t = v8::FunctionTemplate::New(isolate, callback, v8::Local(), s); - v8::Local fn = t->GetFunction(); - recv->PrototypeTemplate()->Set(v8::String::NewFromUtf8(isolate, name), fn); v8::Local fn_name = v8::String::NewFromUtf8(isolate, name); - fn->SetName(fn_name); + t->SetClassName(fn_name); + recv->PrototypeTemplate()->Set(v8::String::NewFromUtf8(isolate, name), t); } #define NODE_SET_PROTOTYPE_METHOD node::NODE_SET_PROTOTYPE_METHOD diff --git a/src/node_http_parser.cc b/src/node_http_parser.cc index 4087ed263fb1a9..7d30cfd99ee19b 100644 --- a/src/node_http_parser.cc +++ b/src/node_http_parser.cc @@ -759,7 +759,7 @@ void InitHttpParser(Local target, methods->Set(num, FIXED_ONE_BYTE_STRING(env->isolate(), #string)); HTTP_METHOD_MAP(V) #undef V - t->Set(FIXED_ONE_BYTE_STRING(env->isolate(), "methods"), methods); + target->Set(FIXED_ONE_BYTE_STRING(env->isolate(), "methods"), methods); env->SetProtoMethod(t, "close", Parser::Close); env->SetProtoMethod(t, "execute", Parser::Execute); diff --git a/test/parallel/test-http-parser.js b/test/parallel/test-http-parser.js index 6c1466c4184a67..b5bf275a6dfc8a 100644 --- a/test/parallel/test-http-parser.js +++ b/test/parallel/test-http-parser.js @@ -2,14 +2,14 @@ require('../common'); var assert = require('assert'); -var HTTPParser = process.binding('http_parser').HTTPParser; +const binding = process.binding('http_parser'); +const methods = binding.methods; +const HTTPParser = binding.HTTPParser; var CRLF = '\r\n'; var REQUEST = HTTPParser.REQUEST; var RESPONSE = HTTPParser.RESPONSE; -var methods = HTTPParser.methods; - var kOnHeaders = HTTPParser.kOnHeaders | 0; var kOnHeadersComplete = HTTPParser.kOnHeadersComplete | 0; var kOnBody = HTTPParser.kOnBody | 0;