diff options
author | Fedor Indutny <fedor@indutny.com> | 2015-10-05 23:26:13 -0400 |
---|---|---|
committer | Fedor Indutny <fedor@indutny.com> | 2015-10-06 18:57:46 -0400 |
commit | d1f24044b9b594af2fc58c8a8812035a50c53bd1 (patch) | |
tree | 2214d3148e5cd5cf740cef978ae35c4570a20cf0 /test | |
parent | 3de353b5544ec5863dc4e1291cdf1aed01b50358 (diff) | |
download | node-new-d1f24044b9b594af2fc58c8a8812035a50c53bd1.tar.gz |
buffer: FreeCallback should be tied to ArrayBuffer
FreeCallback should be invoked on the storage disposal (`ArrayBuffer`),
not when the view (`Uint8Array` or `Buffer`) is disposed. This causes
bug and crashes in addons which create buffers and store only slices of
them.
PR-URL: https://github.com/nodejs/node/pull/3198
Reviewed-By: Ben Noordhuis <info@bnoordhuis.nl>
Reviewed-By: Trevor Norris <trev.norris@gmail.com>
Diffstat (limited to 'test')
-rw-r--r-- | test/addons/buffer-free-callback/binding.cc | 36 | ||||
-rw-r--r-- | test/addons/buffer-free-callback/binding.gyp | 8 | ||||
-rw-r--r-- | test/addons/buffer-free-callback/test.js | 9 |
3 files changed, 53 insertions, 0 deletions
diff --git a/test/addons/buffer-free-callback/binding.cc b/test/addons/buffer-free-callback/binding.cc new file mode 100644 index 0000000000..a1e7773e95 --- /dev/null +++ b/test/addons/buffer-free-callback/binding.cc @@ -0,0 +1,36 @@ +#include <node.h> +#include <node_buffer.h> +#include <util.h> +#include <v8.h> + +static int alive; +static char buf[1024]; + +static void FreeCallback(char* data, void* hint) { + alive--; +} + +void Alloc(const v8::FunctionCallbackInfo<v8::Value>& args) { + v8::Isolate* isolate = args.GetIsolate(); + alive++; + args.GetReturnValue().Set(node::Buffer::New( + isolate, + buf, + sizeof(buf), + FreeCallback, + nullptr).ToLocalChecked()); +} + +void Check(const v8::FunctionCallbackInfo<v8::Value>& args) { + v8::Isolate* isolate = args.GetIsolate(); + isolate->RequestGarbageCollectionForTesting( + v8::Isolate::kFullGarbageCollection); + CHECK_GT(alive, 0); +} + +void init(v8::Local<v8::Object> target) { + NODE_SET_METHOD(target, "alloc", Alloc); + NODE_SET_METHOD(target, "check", Check); +} + +NODE_MODULE(binding, init); diff --git a/test/addons/buffer-free-callback/binding.gyp b/test/addons/buffer-free-callback/binding.gyp new file mode 100644 index 0000000000..3bfb84493f --- /dev/null +++ b/test/addons/buffer-free-callback/binding.gyp @@ -0,0 +1,8 @@ +{ + 'targets': [ + { + 'target_name': 'binding', + 'sources': [ 'binding.cc' ] + } + ] +} diff --git a/test/addons/buffer-free-callback/test.js b/test/addons/buffer-free-callback/test.js new file mode 100644 index 0000000000..14bc350738 --- /dev/null +++ b/test/addons/buffer-free-callback/test.js @@ -0,0 +1,9 @@ +'use strict'; +// Flags: --expose-gc + +var assert = require('assert'); +var binding = require('./build/Release/binding'); +var buf = binding.alloc(); +var slice = buf.slice(32); +buf = null; +binding.check(slice); |