Skip to content

Commit 73fc440

Browse files
committed
buffer: fix buffer alignment restriction
Recent phantom weakness API changes to buffer, ebbbc5a, ending up introducing an alignment restriction on the native buffer pointers. It turns out that there are uses in the modules ecosystem that rely on the ability to create buffers with unaligned pointers (e.g. node-ffi). It turns out there is a simpler solution possible here. As a side effect this also removes the need to have to reserve the first internal field on buffers. PR-URL: nodejs#5752 Reviewed-By: trevnorris - Trevor Norris <[email protected]> Reviewed-By: bnoordhuis - Ben Noordhuis <[email protected]>
1 parent 5f1eb43 commit 73fc440

File tree

2 files changed

+15
-18
lines changed

2 files changed

+15
-18
lines changed

src/node_buffer.cc

+15-14
Original file line numberDiff line numberDiff line change
@@ -87,17 +87,20 @@ class CallbackInfo {
8787
static inline CallbackInfo* New(Isolate* isolate,
8888
Local<ArrayBuffer> object,
8989
FreeCallback callback,
90+
char* data,
9091
void* hint = 0);
9192
private:
9293
static void WeakCallback(const WeakCallbackInfo<CallbackInfo>&);
93-
inline void WeakCallback(Isolate* isolate, char* const data);
94+
inline void WeakCallback(Isolate* isolate);
9495
inline CallbackInfo(Isolate* isolate,
9596
Local<ArrayBuffer> object,
9697
FreeCallback callback,
98+
char* data,
9799
void* hint);
98100
~CallbackInfo();
99101
Persistent<ArrayBuffer> persistent_;
100102
FreeCallback const callback_;
103+
char* const data_;
101104
void* const hint_;
102105
DISALLOW_COPY_AND_ASSIGN(CallbackInfo);
103106
};
@@ -111,27 +114,27 @@ void CallbackInfo::Free(char* data, void*) {
111114
CallbackInfo* CallbackInfo::New(Isolate* isolate,
112115
Local<ArrayBuffer> object,
113116
FreeCallback callback,
117+
char* data,
114118
void* hint) {
115-
return new CallbackInfo(isolate, object, callback, hint);
119+
return new CallbackInfo(isolate, object, callback, data, hint);
116120
}
117121

118122

119123
CallbackInfo::CallbackInfo(Isolate* isolate,
120124
Local<ArrayBuffer> object,
121125
FreeCallback callback,
126+
char* data,
122127
void* hint)
123128
: persistent_(isolate, object),
124129
callback_(callback),
130+
data_(data),
125131
hint_(hint) {
126132
ArrayBuffer::Contents obj_c = object->GetContents();
127-
char* const data = static_cast<char*>(obj_c.Data());
133+
CHECK_EQ(data_, static_cast<char*>(obj_c.Data()));
128134
if (object->ByteLength() != 0)
129-
CHECK_NE(data, nullptr);
130-
131-
object->SetAlignedPointerInInternalField(kBufferInternalFieldIndex, data);
135+
CHECK_NE(data_, nullptr);
132136

133-
persistent_.SetWeak(this, WeakCallback,
134-
v8::WeakCallbackType::kInternalFields);
137+
persistent_.SetWeak(this, WeakCallback, v8::WeakCallbackType::kParameter);
135138
persistent_.SetWrapperClassId(BUFFER_ID);
136139
persistent_.MarkIndependent();
137140
isolate->AdjustAmountOfExternalAllocatedMemory(sizeof(*this));
@@ -146,15 +149,13 @@ CallbackInfo::~CallbackInfo() {
146149
void CallbackInfo::WeakCallback(
147150
const WeakCallbackInfo<CallbackInfo>& data) {
148151
CallbackInfo* self = data.GetParameter();
149-
self->WeakCallback(
150-
data.GetIsolate(),
151-
static_cast<char*>(data.GetInternalField(kBufferInternalFieldIndex)));
152+
self->WeakCallback(data.GetIsolate());
152153
delete self;
153154
}
154155

155156

156-
void CallbackInfo::WeakCallback(Isolate* isolate, char* const data) {
157-
callback_(data, hint_);
157+
void CallbackInfo::WeakCallback(Isolate* isolate) {
158+
callback_(data_, hint_);
158159
int64_t change_in_bytes = -static_cast<int64_t>(sizeof(*this));
159160
isolate->AdjustAmountOfExternalAllocatedMemory(change_in_bytes);
160161
}
@@ -370,7 +371,7 @@ MaybeLocal<Object> New(Environment* env,
370371
if (!mb.FromMaybe(false))
371372
return Local<Object>();
372373

373-
CallbackInfo::New(env->isolate(), ab, callback, hint);
374+
CallbackInfo::New(env->isolate(), ab, callback, data, hint);
374375
return scope.Escape(ui);
375376
}
376377

src/node_buffer.h

-4
Original file line numberDiff line numberDiff line change
@@ -13,10 +13,6 @@ namespace Buffer {
1313
static const unsigned int kMaxLength =
1414
sizeof(int32_t) == sizeof(intptr_t) ? 0x3fffffff : 0x7fffffff;
1515

16-
// Buffers have two internal fields, the first of which is reserved for use by
17-
// Node.
18-
static const unsigned int kBufferInternalFieldIndex = 0;
19-
2016
NODE_EXTERN typedef void (*FreeCallback)(char* data, void* hint);
2117

2218
NODE_EXTERN bool HasInstance(v8::Local<v8::Value> val);

0 commit comments

Comments
 (0)