diff --git a/src/udp_wrap.cc b/src/udp_wrap.cc index 10a5fd929f6037..e60a2162d1f797 100644 --- a/src/udp_wrap.cc +++ b/src/udp_wrap.cc @@ -31,12 +31,16 @@ namespace node { using v8::Array; +using v8::ArrayBuffer; +using v8::BackingStore; +using v8::Boolean; using v8::Context; using v8::DontDelete; using v8::FunctionCallbackInfo; using v8::FunctionTemplate; using v8::HandleScope; using v8::Integer; +using v8::Isolate; using v8::Local; using v8::MaybeLocal; using v8::Object; @@ -44,6 +48,7 @@ using v8::PropertyAttribute; using v8::ReadOnly; using v8::Signature; using v8::Uint32; +using v8::Uint8Array; using v8::Undefined; using v8::Value; @@ -314,7 +319,7 @@ void UDPWrap::BufferSize(const FunctionCallbackInfo& args) { CHECK(args[0]->IsUint32()); CHECK(args[1]->IsBoolean()); - bool is_recv = args[1].As()->Value(); + bool is_recv = args[1].As()->Value(); const char* uv_func_name = is_recv ? "uv_recv_buffer_size" : "uv_send_buffer_size"; @@ -679,7 +684,16 @@ void UDPWrap::OnAlloc(uv_handle_t* handle, } uv_buf_t UDPWrap::OnAlloc(size_t suggested_size) { - return AllocatedBuffer::AllocateManaged(env(), suggested_size).release(); + Environment* env = this->env(); + uv_buf_t buf; + { + NoArrayBufferZeroFillScope no_zero_fill_scope(env->isolate_data()); + std::unique_ptr bs = + ArrayBuffer::NewBackingStore(env->isolate(), suggested_size); + buf = uv_buf_init(static_cast(bs->Data()), bs->ByteLength()); + env->released_allocated_buffers()->emplace(buf.base, std::move(bs)); + } + return buf; } void UDPWrap::OnRecv(uv_udp_t* handle, @@ -696,27 +710,42 @@ void UDPWrap::OnRecv(ssize_t nread, const sockaddr* addr, unsigned int flags) { Environment* env = this->env(); - AllocatedBuffer buf(env, buf_); + Isolate* isolate = env->isolate(); + + std::unique_ptr bs; + if (buf_.base != nullptr) { + auto map = env->released_allocated_buffers(); + auto it = map->find(buf_.base); + CHECK_NE(it, map->end()); + bs = std::move(it->second); + map->erase(it); + } + if (nread == 0 && addr == nullptr) { return; } - HandleScope handle_scope(env->isolate()); + HandleScope handle_scope(isolate); Context::Scope context_scope(env->context()); Local argv[] = { - Integer::New(env->isolate(), static_cast(nread)), + Integer::New(isolate, static_cast(nread)), object(), - Undefined(env->isolate()), - Undefined(env->isolate())}; + Undefined(isolate), + Undefined(isolate)}; if (nread < 0) { MakeCallback(env->onmessage_string(), arraysize(argv), argv); return; + } else if (nread == 0) { + bs = ArrayBuffer::NewBackingStore(isolate, 0); + } else { + NoArrayBufferZeroFillScope no_zero_fill_scope(env->isolate_data()); + bs = BackingStore::Reallocate(isolate, std::move(bs), nread); } - buf.Resize(nread); - argv[2] = buf.ToBuffer().ToLocalChecked(); + Local ab = ArrayBuffer::New(isolate, std::move(bs)); + argv[2] = Buffer::New(env, ab, 0, ab->ByteLength()).ToLocalChecked(); argv[3] = AddressToJS(env, addr); MakeCallback(env->onmessage_string(), arraysize(argv), argv); }