1 // Copyright Joyent, Inc. and other Node contributors.
2 //
3 // Permission is hereby granted, free of charge, to any person obtaining a
4 // copy of this software and associated documentation files (the
5 // "Software"), to deal in the Software without restriction, including
6 // without limitation the rights to use, copy, modify, merge, publish,
7 // distribute, sublicense, and/or sell copies of the Software, and to permit
8 // persons to whom the Software is furnished to do so, subject to the
9 // following conditions:
10 //
11 // The above copyright notice and this permission notice shall be included
12 // in all copies or substantial portions of the Software.
13 //
14 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS
15 // OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
16 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN
17 // NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
18 // DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR
19 // OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE
20 // USE OR OTHER DEALINGS IN THE SOFTWARE.
21
22 #include "memory_tracker-inl.h"
23 #include "node.h"
24 #include "node_buffer.h"
25
26 #include "async_wrap-inl.h"
27 #include "env-inl.h"
28 #include "threadpoolwork-inl.h"
29 #include "util-inl.h"
30
31 #include "v8.h"
32
33 #include "brotli/encode.h"
34 #include "brotli/decode.h"
35 #include "zlib.h"
36
37 #include <sys/types.h>
38
39 #include <cerrno>
40 #include <cstdlib>
41 #include <cstring>
42 #include <atomic>
43
44 namespace node {
45
46 using v8::ArrayBuffer;
47 using v8::Context;
48 using v8::Function;
49 using v8::FunctionCallbackInfo;
50 using v8::FunctionTemplate;
51 using v8::Global;
52 using v8::HandleScope;
53 using v8::Int32;
54 using v8::Integer;
55 using v8::Local;
56 using v8::Object;
57 using v8::Uint32Array;
58 using v8::Value;
59
60 namespace {
61
62 // Fewer than 64 bytes per chunk is not recommended.
63 // Technically it could work with as few as 8, but even 64 bytes
64 // is low. Usually a MB or more is best.
65 #define Z_MIN_CHUNK 64
66 #define Z_MAX_CHUNK std::numeric_limits<double>::infinity()
67 #define Z_DEFAULT_CHUNK (16 * 1024)
68 #define Z_MIN_MEMLEVEL 1
69 #define Z_MAX_MEMLEVEL 9
70 #define Z_DEFAULT_MEMLEVEL 8
71 #define Z_MIN_LEVEL -1
72 #define Z_MAX_LEVEL 9
73 #define Z_DEFAULT_LEVEL Z_DEFAULT_COMPRESSION
74 #define Z_MIN_WINDOWBITS 8
75 #define Z_MAX_WINDOWBITS 15
76 #define Z_DEFAULT_WINDOWBITS 15
77
78 #define ZLIB_ERROR_CODES(V) \
79 V(Z_OK) \
80 V(Z_STREAM_END) \
81 V(Z_NEED_DICT) \
82 V(Z_ERRNO) \
83 V(Z_STREAM_ERROR) \
84 V(Z_DATA_ERROR) \
85 V(Z_MEM_ERROR) \
86 V(Z_BUF_ERROR) \
87 V(Z_VERSION_ERROR) \
88
ZlibStrerror(int err)89 inline const char* ZlibStrerror(int err) {
90 #define V(code) if (err == code) return #code;
91 ZLIB_ERROR_CODES(V)
92 #undef V
93 return "Z_UNKNOWN_ERROR";
94 }
95
96 enum node_zlib_mode {
97 NONE,
98 DEFLATE,
99 INFLATE,
100 GZIP,
101 GUNZIP,
102 DEFLATERAW,
103 INFLATERAW,
104 UNZIP,
105 BROTLI_DECODE,
106 BROTLI_ENCODE
107 };
108
109 #define GZIP_HEADER_ID1 0x1f
110 #define GZIP_HEADER_ID2 0x8b
111
112 struct CompressionError {
CompressionErrornode::__anonb47e24fa0111::CompressionError113 CompressionError(const char* message, const char* code, int err)
114 : message(message),
115 code(code),
116 err(err) {
117 CHECK_NOT_NULL(message);
118 }
119
120 CompressionError() = default;
121
122 const char* message = nullptr;
123 const char* code = nullptr;
124 int err = 0;
125
IsErrornode::__anonb47e24fa0111::CompressionError126 inline bool IsError() const { return code != nullptr; }
127 };
128
129 class ZlibContext : public MemoryRetainer {
130 public:
131 ZlibContext() = default;
132
133 // Streaming-related, should be available for all compression libraries:
134 void Close();
135 void DoThreadPoolWork();
136 void SetBuffers(char* in, uint32_t in_len, char* out, uint32_t out_len);
137 void SetFlush(int flush);
138 void GetAfterWriteOffsets(uint32_t* avail_in, uint32_t* avail_out) const;
139 CompressionError GetErrorInfo() const;
SetMode(node_zlib_mode mode)140 inline void SetMode(node_zlib_mode mode) { mode_ = mode; }
141 CompressionError ResetStream();
142
143 // Zlib-specific:
144 void Init(int level, int window_bits, int mem_level, int strategy,
145 std::vector<unsigned char>&& dictionary);
146 void SetAllocationFunctions(alloc_func alloc, free_func free, void* opaque);
147 CompressionError SetParams(int level, int strategy);
148
149 SET_MEMORY_INFO_NAME(ZlibContext)
SET_SELF_SIZE(ZlibContext)150 SET_SELF_SIZE(ZlibContext)
151
152 void MemoryInfo(MemoryTracker* tracker) const override {
153 tracker->TrackField("dictionary", dictionary_);
154 }
155
156 ZlibContext(const ZlibContext&) = delete;
157 ZlibContext& operator=(const ZlibContext&) = delete;
158
159 private:
160 CompressionError ErrorForMessage(const char* message) const;
161 CompressionError SetDictionary();
162 bool InitZlib();
163
164 Mutex mutex_; // Protects zlib_init_done_.
165 bool zlib_init_done_ = false;
166 int err_ = 0;
167 int flush_ = 0;
168 int level_ = 0;
169 int mem_level_ = 0;
170 node_zlib_mode mode_ = NONE;
171 int strategy_ = 0;
172 int window_bits_ = 0;
173 unsigned int gzip_id_bytes_read_ = 0;
174 std::vector<unsigned char> dictionary_;
175
176 z_stream strm_;
177 };
178
179 // Brotli has different data types for compression and decompression streams,
180 // so some of the specifics are implemented in more specific subclasses
181 class BrotliContext : public MemoryRetainer {
182 public:
183 BrotliContext() = default;
184
185 void SetBuffers(char* in, uint32_t in_len, char* out, uint32_t out_len);
186 void SetFlush(int flush);
187 void GetAfterWriteOffsets(uint32_t* avail_in, uint32_t* avail_out) const;
SetMode(node_zlib_mode mode)188 inline void SetMode(node_zlib_mode mode) { mode_ = mode; }
189
190 BrotliContext(const BrotliContext&) = delete;
191 BrotliContext& operator=(const BrotliContext&) = delete;
192
193 protected:
194 node_zlib_mode mode_ = NONE;
195 uint8_t* next_in_ = nullptr;
196 uint8_t* next_out_ = nullptr;
197 size_t avail_in_ = 0;
198 size_t avail_out_ = 0;
199 BrotliEncoderOperation flush_ = BROTLI_OPERATION_PROCESS;
200 // TODO(addaleax): These should not need to be stored here.
201 // This is currently only done this way to make implementing ResetStream()
202 // easier.
203 brotli_alloc_func alloc_ = nullptr;
204 brotli_free_func free_ = nullptr;
205 void* alloc_opaque_ = nullptr;
206 };
207
208 class BrotliEncoderContext final : public BrotliContext {
209 public:
210 void Close();
211 void DoThreadPoolWork();
212 CompressionError Init(brotli_alloc_func alloc,
213 brotli_free_func free,
214 void* opaque);
215 CompressionError ResetStream();
216 CompressionError SetParams(int key, uint32_t value);
217 CompressionError GetErrorInfo() const;
218
219 SET_MEMORY_INFO_NAME(BrotliEncoderContext)
220 SET_SELF_SIZE(BrotliEncoderContext)
221 SET_NO_MEMORY_INFO() // state_ is covered through allocation tracking.
222
223 private:
224 bool last_result_ = false;
225 DeleteFnPtr<BrotliEncoderState, BrotliEncoderDestroyInstance> state_;
226 };
227
228 class BrotliDecoderContext final : public BrotliContext {
229 public:
230 void Close();
231 void DoThreadPoolWork();
232 CompressionError Init(brotli_alloc_func alloc,
233 brotli_free_func free,
234 void* opaque);
235 CompressionError ResetStream();
236 CompressionError SetParams(int key, uint32_t value);
237 CompressionError GetErrorInfo() const;
238
239 SET_MEMORY_INFO_NAME(BrotliDecoderContext)
240 SET_SELF_SIZE(BrotliDecoderContext)
241 SET_NO_MEMORY_INFO() // state_ is covered through allocation tracking.
242
243 private:
244 BrotliDecoderResult last_result_ = BROTLI_DECODER_RESULT_SUCCESS;
245 BrotliDecoderErrorCode error_ = BROTLI_DECODER_NO_ERROR;
246 std::string error_string_;
247 DeleteFnPtr<BrotliDecoderState, BrotliDecoderDestroyInstance> state_;
248 };
249
250 template <typename CompressionContext>
251 class CompressionStream : public AsyncWrap, public ThreadPoolWork {
252 public:
CompressionStream(Environment * env,Local<Object> wrap)253 CompressionStream(Environment* env, Local<Object> wrap)
254 : AsyncWrap(env, wrap, AsyncWrap::PROVIDER_ZLIB),
255 ThreadPoolWork(env),
256 write_result_(nullptr) {
257 MakeWeak();
258 }
259
~CompressionStream()260 ~CompressionStream() override {
261 CHECK_EQ(false, write_in_progress_ && "write in progress");
262 Close();
263 CHECK_EQ(zlib_memory_, 0);
264 CHECK_EQ(unreported_allocations_, 0);
265 }
266
Close()267 void Close() {
268 if (write_in_progress_) {
269 pending_close_ = true;
270 return;
271 }
272
273 pending_close_ = false;
274 closed_ = true;
275 CHECK(init_done_ && "close before init");
276
277 AllocScope alloc_scope(this);
278 ctx_.Close();
279 }
280
281
Close(const FunctionCallbackInfo<Value> & args)282 static void Close(const FunctionCallbackInfo<Value>& args) {
283 CompressionStream* ctx;
284 ASSIGN_OR_RETURN_UNWRAP(&ctx, args.Holder());
285 ctx->Close();
286 }
287
288
289 // write(flush, in, in_off, in_len, out, out_off, out_len)
290 template <bool async>
Write(const FunctionCallbackInfo<Value> & args)291 static void Write(const FunctionCallbackInfo<Value>& args) {
292 Environment* env = Environment::GetCurrent(args);
293 Local<Context> context = env->context();
294 CHECK_EQ(args.Length(), 7);
295
296 uint32_t in_off, in_len, out_off, out_len, flush;
297 char* in;
298 char* out;
299
300 CHECK_EQ(false, args[0]->IsUndefined() && "must provide flush value");
301 if (!args[0]->Uint32Value(context).To(&flush)) return;
302
303 if (flush != Z_NO_FLUSH &&
304 flush != Z_PARTIAL_FLUSH &&
305 flush != Z_SYNC_FLUSH &&
306 flush != Z_FULL_FLUSH &&
307 flush != Z_FINISH &&
308 flush != Z_BLOCK) {
309 CHECK(0 && "Invalid flush value");
310 }
311
312 if (args[1]->IsNull()) {
313 // just a flush
314 in = nullptr;
315 in_len = 0;
316 in_off = 0;
317 } else {
318 CHECK(Buffer::HasInstance(args[1]));
319 Local<Object> in_buf = args[1].As<Object>();
320 if (!args[2]->Uint32Value(context).To(&in_off)) return;
321 if (!args[3]->Uint32Value(context).To(&in_len)) return;
322
323 CHECK(Buffer::IsWithinBounds(in_off, in_len, Buffer::Length(in_buf)));
324 in = Buffer::Data(in_buf) + in_off;
325 }
326
327 CHECK(Buffer::HasInstance(args[4]));
328 Local<Object> out_buf = args[4].As<Object>();
329 if (!args[5]->Uint32Value(context).To(&out_off)) return;
330 if (!args[6]->Uint32Value(context).To(&out_len)) return;
331 CHECK(Buffer::IsWithinBounds(out_off, out_len, Buffer::Length(out_buf)));
332 out = Buffer::Data(out_buf) + out_off;
333
334 CompressionStream* ctx;
335 ASSIGN_OR_RETURN_UNWRAP(&ctx, args.Holder());
336
337 ctx->Write<async>(flush, in, in_len, out, out_len);
338 }
339
340 template <bool async>
Write(uint32_t flush,char * in,uint32_t in_len,char * out,uint32_t out_len)341 void Write(uint32_t flush,
342 char* in, uint32_t in_len,
343 char* out, uint32_t out_len) {
344 AllocScope alloc_scope(this);
345
346 CHECK(init_done_ && "write before init");
347 CHECK(!closed_ && "already finalized");
348
349 CHECK_EQ(false, write_in_progress_);
350 CHECK_EQ(false, pending_close_);
351 write_in_progress_ = true;
352 Ref();
353
354 ctx_.SetBuffers(in, in_len, out, out_len);
355 ctx_.SetFlush(flush);
356
357 if (!async) {
358 // sync version
359 AsyncWrap::env()->PrintSyncTrace();
360 DoThreadPoolWork();
361 if (CheckError()) {
362 UpdateWriteResult();
363 write_in_progress_ = false;
364 }
365 Unref();
366 return;
367 }
368
369 // async version
370 ScheduleWork();
371 }
372
UpdateWriteResult()373 void UpdateWriteResult() {
374 ctx_.GetAfterWriteOffsets(&write_result_[1], &write_result_[0]);
375 }
376
377 // thread pool!
378 // This function may be called multiple times on the uv_work pool
379 // for a single write() call, until all of the input bytes have
380 // been consumed.
DoThreadPoolWork()381 void DoThreadPoolWork() override {
382 ctx_.DoThreadPoolWork();
383 }
384
385
CheckError()386 bool CheckError() {
387 const CompressionError err = ctx_.GetErrorInfo();
388 if (!err.IsError()) return true;
389 EmitError(err);
390 return false;
391 }
392
393
394 // v8 land!
AfterThreadPoolWork(int status)395 void AfterThreadPoolWork(int status) override {
396 AllocScope alloc_scope(this);
397 auto on_scope_leave = OnScopeLeave([&]() { Unref(); });
398
399 write_in_progress_ = false;
400
401 if (status == UV_ECANCELED) {
402 Close();
403 return;
404 }
405
406 CHECK_EQ(status, 0);
407
408 Environment* env = AsyncWrap::env();
409 HandleScope handle_scope(env->isolate());
410 Context::Scope context_scope(env->context());
411
412 if (!CheckError())
413 return;
414
415 UpdateWriteResult();
416
417 // call the write() cb
418 Local<Function> cb = PersistentToLocal::Default(env->isolate(),
419 write_js_callback_);
420 MakeCallback(cb, 0, nullptr);
421
422 if (pending_close_)
423 Close();
424 }
425
426 // TODO(addaleax): Switch to modern error system (node_errors.h).
EmitError(const CompressionError & err)427 void EmitError(const CompressionError& err) {
428 Environment* env = AsyncWrap::env();
429 // If you hit this assertion, you forgot to enter the v8::Context first.
430 CHECK_EQ(env->context(), env->isolate()->GetCurrentContext());
431
432 HandleScope scope(env->isolate());
433 Local<Value> args[3] = {
434 OneByteString(env->isolate(), err.message),
435 Integer::New(env->isolate(), err.err),
436 OneByteString(env->isolate(), err.code)
437 };
438 MakeCallback(env->onerror_string(), arraysize(args), args);
439
440 // no hope of rescue.
441 write_in_progress_ = false;
442 if (pending_close_)
443 Close();
444 }
445
Reset(const FunctionCallbackInfo<Value> & args)446 static void Reset(const FunctionCallbackInfo<Value> &args) {
447 CompressionStream* wrap;
448 ASSIGN_OR_RETURN_UNWRAP(&wrap, args.Holder());
449
450 AllocScope alloc_scope(wrap);
451 const CompressionError err = wrap->context()->ResetStream();
452 if (err.IsError())
453 wrap->EmitError(err);
454 }
455
MemoryInfo(MemoryTracker * tracker) const456 void MemoryInfo(MemoryTracker* tracker) const override {
457 tracker->TrackField("compression context", ctx_);
458 tracker->TrackFieldWithSize("zlib_memory",
459 zlib_memory_ + unreported_allocations_);
460 }
461
462 protected:
context()463 CompressionContext* context() { return &ctx_; }
464
InitStream(uint32_t * write_result,Local<Function> write_js_callback)465 void InitStream(uint32_t* write_result, Local<Function> write_js_callback) {
466 write_result_ = write_result;
467 write_js_callback_.Reset(AsyncWrap::env()->isolate(), write_js_callback);
468 init_done_ = true;
469 }
470
471 // Allocation functions provided to zlib itself. We store the real size of
472 // the allocated memory chunk just before the "payload" memory we return
473 // to zlib.
474 // Because we use zlib off the thread pool, we can not report memory directly
475 // to V8; rather, we first store it as "unreported" memory in a separate
476 // field and later report it back from the main thread.
AllocForZlib(void * data,uInt items,uInt size)477 static void* AllocForZlib(void* data, uInt items, uInt size) {
478 size_t real_size =
479 MultiplyWithOverflowCheck(static_cast<size_t>(items),
480 static_cast<size_t>(size));
481 return AllocForBrotli(data, real_size);
482 }
483
AllocForBrotli(void * data,size_t size)484 static void* AllocForBrotli(void* data, size_t size) {
485 size += sizeof(size_t);
486 CompressionStream* ctx = static_cast<CompressionStream*>(data);
487 char* memory = UncheckedMalloc(size);
488 if (UNLIKELY(memory == nullptr)) return nullptr;
489 *reinterpret_cast<size_t*>(memory) = size;
490 ctx->unreported_allocations_.fetch_add(size,
491 std::memory_order_relaxed);
492 return memory + sizeof(size_t);
493 }
494
FreeForZlib(void * data,void * pointer)495 static void FreeForZlib(void* data, void* pointer) {
496 if (UNLIKELY(pointer == nullptr)) return;
497 CompressionStream* ctx = static_cast<CompressionStream*>(data);
498 char* real_pointer = static_cast<char*>(pointer) - sizeof(size_t);
499 size_t real_size = *reinterpret_cast<size_t*>(real_pointer);
500 ctx->unreported_allocations_.fetch_sub(real_size,
501 std::memory_order_relaxed);
502 free(real_pointer);
503 }
504
505 // This is called on the main thread after zlib may have allocated something
506 // in order to report it back to V8.
AdjustAmountOfExternalAllocatedMemory()507 void AdjustAmountOfExternalAllocatedMemory() {
508 ssize_t report =
509 unreported_allocations_.exchange(0, std::memory_order_relaxed);
510 if (report == 0) return;
511 CHECK_IMPLIES(report < 0, zlib_memory_ >= static_cast<size_t>(-report));
512 zlib_memory_ += report;
513 AsyncWrap::env()->isolate()->AdjustAmountOfExternalAllocatedMemory(report);
514 }
515
516 struct AllocScope {
AllocScopenode::__anonb47e24fa0111::CompressionStream::AllocScope517 explicit AllocScope(CompressionStream* stream) : stream(stream) {}
~AllocScopenode::__anonb47e24fa0111::CompressionStream::AllocScope518 ~AllocScope() { stream->AdjustAmountOfExternalAllocatedMemory(); }
519 CompressionStream* stream;
520 };
521
522 private:
Ref()523 void Ref() {
524 if (++refs_ == 1) {
525 ClearWeak();
526 }
527 }
528
Unref()529 void Unref() {
530 CHECK_GT(refs_, 0);
531 if (--refs_ == 0) {
532 MakeWeak();
533 }
534 }
535
536 bool init_done_ = false;
537 bool write_in_progress_ = false;
538 bool pending_close_ = false;
539 bool closed_ = false;
540 unsigned int refs_ = 0;
541 uint32_t* write_result_ = nullptr;
542 Global<Function> write_js_callback_;
543 std::atomic<ssize_t> unreported_allocations_{0};
544 size_t zlib_memory_ = 0;
545
546 CompressionContext ctx_;
547 };
548
549 class ZlibStream : public CompressionStream<ZlibContext> {
550 public:
ZlibStream(Environment * env,Local<Object> wrap,node_zlib_mode mode)551 ZlibStream(Environment* env, Local<Object> wrap, node_zlib_mode mode)
552 : CompressionStream(env, wrap) {
553 context()->SetMode(mode);
554 }
555
New(const FunctionCallbackInfo<Value> & args)556 static void New(const FunctionCallbackInfo<Value>& args) {
557 Environment* env = Environment::GetCurrent(args);
558 CHECK(args[0]->IsInt32());
559 node_zlib_mode mode =
560 static_cast<node_zlib_mode>(args[0].As<Int32>()->Value());
561 new ZlibStream(env, args.This(), mode);
562 }
563
564 // just pull the ints out of the args and call the other Init
Init(const FunctionCallbackInfo<Value> & args)565 static void Init(const FunctionCallbackInfo<Value>& args) {
566 // Refs: https://github.com/nodejs/node/issues/16649
567 // Refs: https://github.com/nodejs/node/issues/14161
568 if (args.Length() == 5) {
569 fprintf(stderr,
570 "WARNING: You are likely using a version of node-tar or npm that "
571 "is incompatible with this version of Node.js.\nPlease use "
572 "either the version of npm that is bundled with Node.js, or "
573 "a version of npm (> 5.5.1 or < 5.4.0) or node-tar (> 4.0.1) "
574 "that is compatible with Node.js 9 and above.\n");
575 }
576 CHECK(args.Length() == 7 &&
577 "init(windowBits, level, memLevel, strategy, writeResult, writeCallback,"
578 " dictionary)");
579
580 ZlibStream* wrap;
581 ASSIGN_OR_RETURN_UNWRAP(&wrap, args.Holder());
582
583 Local<Context> context = args.GetIsolate()->GetCurrentContext();
584
585 // windowBits is special. On the compression side, 0 is an invalid value.
586 // But on the decompression side, a value of 0 for windowBits tells zlib
587 // to use the window size in the zlib header of the compressed stream.
588 uint32_t window_bits;
589 if (!args[0]->Uint32Value(context).To(&window_bits)) return;
590
591 int32_t level;
592 if (!args[1]->Int32Value(context).To(&level)) return;
593
594 uint32_t mem_level;
595 if (!args[2]->Uint32Value(context).To(&mem_level)) return;
596
597 uint32_t strategy;
598 if (!args[3]->Uint32Value(context).To(&strategy)) return;
599
600 CHECK(args[4]->IsUint32Array());
601 Local<Uint32Array> array = args[4].As<Uint32Array>();
602 Local<ArrayBuffer> ab = array->Buffer();
603 uint32_t* write_result = static_cast<uint32_t*>(
604 ab->GetBackingStore()->Data());
605
606 CHECK(args[5]->IsFunction());
607 Local<Function> write_js_callback = args[5].As<Function>();
608
609 std::vector<unsigned char> dictionary;
610 if (Buffer::HasInstance(args[6])) {
611 unsigned char* data =
612 reinterpret_cast<unsigned char*>(Buffer::Data(args[6]));
613 dictionary = std::vector<unsigned char>(
614 data,
615 data + Buffer::Length(args[6]));
616 }
617
618 wrap->InitStream(write_result, write_js_callback);
619
620 AllocScope alloc_scope(wrap);
621 wrap->context()->SetAllocationFunctions(
622 AllocForZlib, FreeForZlib, static_cast<CompressionStream*>(wrap));
623 wrap->context()->Init(level, window_bits, mem_level, strategy,
624 std::move(dictionary));
625 }
626
Params(const FunctionCallbackInfo<Value> & args)627 static void Params(const FunctionCallbackInfo<Value>& args) {
628 CHECK(args.Length() == 2 && "params(level, strategy)");
629 ZlibStream* wrap;
630 ASSIGN_OR_RETURN_UNWRAP(&wrap, args.Holder());
631 Local<Context> context = args.GetIsolate()->GetCurrentContext();
632 int level;
633 if (!args[0]->Int32Value(context).To(&level)) return;
634 int strategy;
635 if (!args[1]->Int32Value(context).To(&strategy)) return;
636
637 AllocScope alloc_scope(wrap);
638 const CompressionError err = wrap->context()->SetParams(level, strategy);
639 if (err.IsError())
640 wrap->EmitError(err);
641 }
642
643 SET_MEMORY_INFO_NAME(ZlibStream)
644 SET_SELF_SIZE(ZlibStream)
645 };
646
647 template <typename CompressionContext>
648 class BrotliCompressionStream : public CompressionStream<CompressionContext> {
649 public:
BrotliCompressionStream(Environment * env,Local<Object> wrap,node_zlib_mode mode)650 BrotliCompressionStream(Environment* env,
651 Local<Object> wrap,
652 node_zlib_mode mode)
653 : CompressionStream<CompressionContext>(env, wrap) {
654 context()->SetMode(mode);
655 }
656
context()657 inline CompressionContext* context() {
658 return this->CompressionStream<CompressionContext>::context();
659 }
660 typedef typename CompressionStream<CompressionContext>::AllocScope AllocScope;
661
New(const FunctionCallbackInfo<Value> & args)662 static void New(const FunctionCallbackInfo<Value>& args) {
663 Environment* env = Environment::GetCurrent(args);
664 CHECK(args[0]->IsInt32());
665 node_zlib_mode mode =
666 static_cast<node_zlib_mode>(args[0].As<Int32>()->Value());
667 new BrotliCompressionStream(env, args.This(), mode);
668 }
669
Init(const FunctionCallbackInfo<Value> & args)670 static void Init(const FunctionCallbackInfo<Value>& args) {
671 BrotliCompressionStream* wrap;
672 ASSIGN_OR_RETURN_UNWRAP(&wrap, args.Holder());
673 CHECK(args.Length() == 3 && "init(params, writeResult, writeCallback)");
674
675 CHECK(args[1]->IsUint32Array());
676 uint32_t* write_result = reinterpret_cast<uint32_t*>(Buffer::Data(args[1]));
677
678 CHECK(args[2]->IsFunction());
679 Local<Function> write_js_callback = args[2].As<Function>();
680 wrap->InitStream(write_result, write_js_callback);
681
682 AllocScope alloc_scope(wrap);
683 CompressionError err =
684 wrap->context()->Init(
685 CompressionStream<CompressionContext>::AllocForBrotli,
686 CompressionStream<CompressionContext>::FreeForZlib,
687 static_cast<CompressionStream<CompressionContext>*>(wrap));
688 if (err.IsError()) {
689 wrap->EmitError(err);
690 args.GetReturnValue().Set(false);
691 return;
692 }
693
694 CHECK(args[0]->IsUint32Array());
695 const uint32_t* data = reinterpret_cast<uint32_t*>(Buffer::Data(args[0]));
696 size_t len = args[0].As<Uint32Array>()->Length();
697
698 for (int i = 0; static_cast<size_t>(i) < len; i++) {
699 if (data[i] == static_cast<uint32_t>(-1))
700 continue;
701 err = wrap->context()->SetParams(i, data[i]);
702 if (err.IsError()) {
703 wrap->EmitError(err);
704 args.GetReturnValue().Set(false);
705 return;
706 }
707 }
708
709 args.GetReturnValue().Set(true);
710 }
711
Params(const FunctionCallbackInfo<Value> & args)712 static void Params(const FunctionCallbackInfo<Value>& args) {
713 // Currently a no-op, and not accessed from JS land.
714 // At some point Brotli may support changing parameters on the fly,
715 // in which case we can implement this and a JS equivalent similar to
716 // the zlib Params() function.
717 }
718
719 SET_MEMORY_INFO_NAME(BrotliCompressionStream)
720 SET_SELF_SIZE(BrotliCompressionStream)
721 };
722
723 using BrotliEncoderStream = BrotliCompressionStream<BrotliEncoderContext>;
724 using BrotliDecoderStream = BrotliCompressionStream<BrotliDecoderContext>;
725
Close()726 void ZlibContext::Close() {
727 {
728 Mutex::ScopedLock lock(mutex_);
729 if (!zlib_init_done_) {
730 dictionary_.clear();
731 mode_ = NONE;
732 return;
733 }
734 }
735
736 CHECK_LE(mode_, UNZIP);
737
738 int status = Z_OK;
739 if (mode_ == DEFLATE || mode_ == GZIP || mode_ == DEFLATERAW) {
740 status = deflateEnd(&strm_);
741 } else if (mode_ == INFLATE || mode_ == GUNZIP || mode_ == INFLATERAW ||
742 mode_ == UNZIP) {
743 status = inflateEnd(&strm_);
744 }
745
746 CHECK(status == Z_OK || status == Z_DATA_ERROR);
747 mode_ = NONE;
748
749 dictionary_.clear();
750 }
751
752
DoThreadPoolWork()753 void ZlibContext::DoThreadPoolWork() {
754 bool first_init_call = InitZlib();
755 if (first_init_call && err_ != Z_OK) {
756 return;
757 }
758
759 const Bytef* next_expected_header_byte = nullptr;
760
761 // If the avail_out is left at 0, then it means that it ran out
762 // of room. If there was avail_out left over, then it means
763 // that all of the input was consumed.
764 switch (mode_) {
765 case DEFLATE:
766 case GZIP:
767 case DEFLATERAW:
768 err_ = deflate(&strm_, flush_);
769 break;
770 case UNZIP:
771 if (strm_.avail_in > 0) {
772 next_expected_header_byte = strm_.next_in;
773 }
774
775 switch (gzip_id_bytes_read_) {
776 case 0:
777 if (next_expected_header_byte == nullptr) {
778 break;
779 }
780
781 if (*next_expected_header_byte == GZIP_HEADER_ID1) {
782 gzip_id_bytes_read_ = 1;
783 next_expected_header_byte++;
784
785 if (strm_.avail_in == 1) {
786 // The only available byte was already read.
787 break;
788 }
789 } else {
790 mode_ = INFLATE;
791 break;
792 }
793
794 // fallthrough
795 case 1:
796 if (next_expected_header_byte == nullptr) {
797 break;
798 }
799
800 if (*next_expected_header_byte == GZIP_HEADER_ID2) {
801 gzip_id_bytes_read_ = 2;
802 mode_ = GUNZIP;
803 } else {
804 // There is no actual difference between INFLATE and INFLATERAW
805 // (after initialization).
806 mode_ = INFLATE;
807 }
808
809 break;
810 default:
811 CHECK(0 && "invalid number of gzip magic number bytes read");
812 }
813
814 // fallthrough
815 case INFLATE:
816 case GUNZIP:
817 case INFLATERAW:
818 err_ = inflate(&strm_, flush_);
819
820 // If data was encoded with dictionary (INFLATERAW will have it set in
821 // SetDictionary, don't repeat that here)
822 if (mode_ != INFLATERAW &&
823 err_ == Z_NEED_DICT &&
824 !dictionary_.empty()) {
825 // Load it
826 err_ = inflateSetDictionary(&strm_,
827 dictionary_.data(),
828 dictionary_.size());
829 if (err_ == Z_OK) {
830 // And try to decode again
831 err_ = inflate(&strm_, flush_);
832 } else if (err_ == Z_DATA_ERROR) {
833 // Both inflateSetDictionary() and inflate() return Z_DATA_ERROR.
834 // Make it possible for After() to tell a bad dictionary from bad
835 // input.
836 err_ = Z_NEED_DICT;
837 }
838 }
839
840 while (strm_.avail_in > 0 &&
841 mode_ == GUNZIP &&
842 err_ == Z_STREAM_END &&
843 strm_.next_in[0] != 0x00) {
844 // Bytes remain in input buffer. Perhaps this is another compressed
845 // member in the same archive, or just trailing garbage.
846 // Trailing zero bytes are okay, though, since they are frequently
847 // used for padding.
848
849 ResetStream();
850 err_ = inflate(&strm_, flush_);
851 }
852 break;
853 default:
854 UNREACHABLE();
855 }
856 }
857
858
SetBuffers(char * in,uint32_t in_len,char * out,uint32_t out_len)859 void ZlibContext::SetBuffers(char* in, uint32_t in_len,
860 char* out, uint32_t out_len) {
861 strm_.avail_in = in_len;
862 strm_.next_in = reinterpret_cast<Bytef*>(in);
863 strm_.avail_out = out_len;
864 strm_.next_out = reinterpret_cast<Bytef*>(out);
865 }
866
867
SetFlush(int flush)868 void ZlibContext::SetFlush(int flush) {
869 flush_ = flush;
870 }
871
872
GetAfterWriteOffsets(uint32_t * avail_in,uint32_t * avail_out) const873 void ZlibContext::GetAfterWriteOffsets(uint32_t* avail_in,
874 uint32_t* avail_out) const {
875 *avail_in = strm_.avail_in;
876 *avail_out = strm_.avail_out;
877 }
878
879
ErrorForMessage(const char * message) const880 CompressionError ZlibContext::ErrorForMessage(const char* message) const {
881 if (strm_.msg != nullptr)
882 message = strm_.msg;
883
884 return CompressionError { message, ZlibStrerror(err_), err_ };
885 }
886
887
GetErrorInfo() const888 CompressionError ZlibContext::GetErrorInfo() const {
889 // Acceptable error states depend on the type of zlib stream.
890 switch (err_) {
891 case Z_OK:
892 case Z_BUF_ERROR:
893 if (strm_.avail_out != 0 && flush_ == Z_FINISH) {
894 return ErrorForMessage("unexpected end of file");
895 }
896 case Z_STREAM_END:
897 // normal statuses, not fatal
898 break;
899 case Z_NEED_DICT:
900 if (dictionary_.empty())
901 return ErrorForMessage("Missing dictionary");
902 else
903 return ErrorForMessage("Bad dictionary");
904 default:
905 // something else.
906 return ErrorForMessage("Zlib error");
907 }
908
909 return CompressionError {};
910 }
911
912
ResetStream()913 CompressionError ZlibContext::ResetStream() {
914 bool first_init_call = InitZlib();
915 if (first_init_call && err_ != Z_OK) {
916 return ErrorForMessage("Failed to init stream before reset");
917 }
918
919 err_ = Z_OK;
920
921 switch (mode_) {
922 case DEFLATE:
923 case DEFLATERAW:
924 case GZIP:
925 err_ = deflateReset(&strm_);
926 break;
927 case INFLATE:
928 case INFLATERAW:
929 case GUNZIP:
930 err_ = inflateReset(&strm_);
931 break;
932 default:
933 break;
934 }
935
936 if (err_ != Z_OK)
937 return ErrorForMessage("Failed to reset stream");
938
939 return SetDictionary();
940 }
941
942
SetAllocationFunctions(alloc_func alloc,free_func free,void * opaque)943 void ZlibContext::SetAllocationFunctions(alloc_func alloc,
944 free_func free,
945 void* opaque) {
946 strm_.zalloc = alloc;
947 strm_.zfree = free;
948 strm_.opaque = opaque;
949 }
950
951
Init(int level,int window_bits,int mem_level,int strategy,std::vector<unsigned char> && dictionary)952 void ZlibContext::Init(
953 int level, int window_bits, int mem_level, int strategy,
954 std::vector<unsigned char>&& dictionary) {
955 if (!((window_bits == 0) &&
956 (mode_ == INFLATE ||
957 mode_ == GUNZIP ||
958 mode_ == UNZIP))) {
959 CHECK(
960 (window_bits >= Z_MIN_WINDOWBITS && window_bits <= Z_MAX_WINDOWBITS) &&
961 "invalid windowBits");
962 }
963
964 CHECK((level >= Z_MIN_LEVEL && level <= Z_MAX_LEVEL) &&
965 "invalid compression level");
966
967 CHECK((mem_level >= Z_MIN_MEMLEVEL && mem_level <= Z_MAX_MEMLEVEL) &&
968 "invalid memlevel");
969
970 CHECK((strategy == Z_FILTERED || strategy == Z_HUFFMAN_ONLY ||
971 strategy == Z_RLE || strategy == Z_FIXED ||
972 strategy == Z_DEFAULT_STRATEGY) &&
973 "invalid strategy");
974
975 level_ = level;
976 window_bits_ = window_bits;
977 mem_level_ = mem_level;
978 strategy_ = strategy;
979
980 flush_ = Z_NO_FLUSH;
981
982 err_ = Z_OK;
983
984 if (mode_ == GZIP || mode_ == GUNZIP) {
985 window_bits_ += 16;
986 }
987
988 if (mode_ == UNZIP) {
989 window_bits_ += 32;
990 }
991
992 if (mode_ == DEFLATERAW || mode_ == INFLATERAW) {
993 window_bits_ *= -1;
994 }
995
996 dictionary_ = std::move(dictionary);
997 }
998
InitZlib()999 bool ZlibContext::InitZlib() {
1000 Mutex::ScopedLock lock(mutex_);
1001 if (zlib_init_done_) {
1002 return false;
1003 }
1004
1005 switch (mode_) {
1006 case DEFLATE:
1007 case GZIP:
1008 case DEFLATERAW:
1009 err_ = deflateInit2(&strm_,
1010 level_,
1011 Z_DEFLATED,
1012 window_bits_,
1013 mem_level_,
1014 strategy_);
1015 break;
1016 case INFLATE:
1017 case GUNZIP:
1018 case INFLATERAW:
1019 case UNZIP:
1020 err_ = inflateInit2(&strm_, window_bits_);
1021 break;
1022 default:
1023 UNREACHABLE();
1024 }
1025
1026 if (err_ != Z_OK) {
1027 dictionary_.clear();
1028 mode_ = NONE;
1029 return true;
1030 }
1031
1032 SetDictionary();
1033 zlib_init_done_ = true;
1034 return true;
1035 }
1036
1037
SetDictionary()1038 CompressionError ZlibContext::SetDictionary() {
1039 if (dictionary_.empty())
1040 return CompressionError {};
1041
1042 err_ = Z_OK;
1043
1044 switch (mode_) {
1045 case DEFLATE:
1046 case DEFLATERAW:
1047 err_ = deflateSetDictionary(&strm_,
1048 dictionary_.data(),
1049 dictionary_.size());
1050 break;
1051 case INFLATERAW:
1052 // The other inflate cases will have the dictionary set when inflate()
1053 // returns Z_NEED_DICT in Process()
1054 err_ = inflateSetDictionary(&strm_,
1055 dictionary_.data(),
1056 dictionary_.size());
1057 break;
1058 default:
1059 break;
1060 }
1061
1062 if (err_ != Z_OK) {
1063 return ErrorForMessage("Failed to set dictionary");
1064 }
1065
1066 return CompressionError {};
1067 }
1068
1069
SetParams(int level,int strategy)1070 CompressionError ZlibContext::SetParams(int level, int strategy) {
1071 bool first_init_call = InitZlib();
1072 if (first_init_call && err_ != Z_OK) {
1073 return ErrorForMessage("Failed to init stream before set parameters");
1074 }
1075
1076 err_ = Z_OK;
1077
1078 switch (mode_) {
1079 case DEFLATE:
1080 case DEFLATERAW:
1081 err_ = deflateParams(&strm_, level, strategy);
1082 break;
1083 default:
1084 break;
1085 }
1086
1087 if (err_ != Z_OK && err_ != Z_BUF_ERROR) {
1088 return ErrorForMessage("Failed to set parameters");
1089 }
1090
1091 return CompressionError {};
1092 }
1093
1094
SetBuffers(char * in,uint32_t in_len,char * out,uint32_t out_len)1095 void BrotliContext::SetBuffers(char* in, uint32_t in_len,
1096 char* out, uint32_t out_len) {
1097 next_in_ = reinterpret_cast<uint8_t*>(in);
1098 next_out_ = reinterpret_cast<uint8_t*>(out);
1099 avail_in_ = in_len;
1100 avail_out_ = out_len;
1101 }
1102
1103
SetFlush(int flush)1104 void BrotliContext::SetFlush(int flush) {
1105 flush_ = static_cast<BrotliEncoderOperation>(flush);
1106 }
1107
1108
GetAfterWriteOffsets(uint32_t * avail_in,uint32_t * avail_out) const1109 void BrotliContext::GetAfterWriteOffsets(uint32_t* avail_in,
1110 uint32_t* avail_out) const {
1111 *avail_in = avail_in_;
1112 *avail_out = avail_out_;
1113 }
1114
1115
DoThreadPoolWork()1116 void BrotliEncoderContext::DoThreadPoolWork() {
1117 CHECK_EQ(mode_, BROTLI_ENCODE);
1118 CHECK(state_);
1119 const uint8_t* next_in = next_in_;
1120 last_result_ = BrotliEncoderCompressStream(state_.get(),
1121 flush_,
1122 &avail_in_,
1123 &next_in,
1124 &avail_out_,
1125 &next_out_,
1126 nullptr);
1127 next_in_ += next_in - next_in_;
1128 }
1129
1130
Close()1131 void BrotliEncoderContext::Close() {
1132 state_.reset();
1133 mode_ = NONE;
1134 }
1135
Init(brotli_alloc_func alloc,brotli_free_func free,void * opaque)1136 CompressionError BrotliEncoderContext::Init(brotli_alloc_func alloc,
1137 brotli_free_func free,
1138 void* opaque) {
1139 alloc_ = alloc;
1140 free_ = free;
1141 alloc_opaque_ = opaque;
1142 state_.reset(BrotliEncoderCreateInstance(alloc, free, opaque));
1143 if (!state_) {
1144 return CompressionError("Could not initialize Brotli instance",
1145 "ERR_ZLIB_INITIALIZATION_FAILED",
1146 -1);
1147 } else {
1148 return CompressionError {};
1149 }
1150 }
1151
ResetStream()1152 CompressionError BrotliEncoderContext::ResetStream() {
1153 return Init(alloc_, free_, alloc_opaque_);
1154 }
1155
SetParams(int key,uint32_t value)1156 CompressionError BrotliEncoderContext::SetParams(int key, uint32_t value) {
1157 if (!BrotliEncoderSetParameter(state_.get(),
1158 static_cast<BrotliEncoderParameter>(key),
1159 value)) {
1160 return CompressionError("Setting parameter failed",
1161 "ERR_BROTLI_PARAM_SET_FAILED",
1162 -1);
1163 } else {
1164 return CompressionError {};
1165 }
1166 }
1167
GetErrorInfo() const1168 CompressionError BrotliEncoderContext::GetErrorInfo() const {
1169 if (!last_result_) {
1170 return CompressionError("Compression failed",
1171 "ERR_BROTLI_COMPRESSION_FAILED",
1172 -1);
1173 } else {
1174 return CompressionError {};
1175 }
1176 }
1177
1178
Close()1179 void BrotliDecoderContext::Close() {
1180 state_.reset();
1181 mode_ = NONE;
1182 }
1183
DoThreadPoolWork()1184 void BrotliDecoderContext::DoThreadPoolWork() {
1185 CHECK_EQ(mode_, BROTLI_DECODE);
1186 CHECK(state_);
1187 const uint8_t* next_in = next_in_;
1188 last_result_ = BrotliDecoderDecompressStream(state_.get(),
1189 &avail_in_,
1190 &next_in,
1191 &avail_out_,
1192 &next_out_,
1193 nullptr);
1194 next_in_ += next_in - next_in_;
1195 if (last_result_ == BROTLI_DECODER_RESULT_ERROR) {
1196 error_ = BrotliDecoderGetErrorCode(state_.get());
1197 error_string_ = std::string("ERR_") + BrotliDecoderErrorString(error_);
1198 }
1199 }
1200
Init(brotli_alloc_func alloc,brotli_free_func free,void * opaque)1201 CompressionError BrotliDecoderContext::Init(brotli_alloc_func alloc,
1202 brotli_free_func free,
1203 void* opaque) {
1204 alloc_ = alloc;
1205 free_ = free;
1206 alloc_opaque_ = opaque;
1207 state_.reset(BrotliDecoderCreateInstance(alloc, free, opaque));
1208 if (!state_) {
1209 return CompressionError("Could not initialize Brotli instance",
1210 "ERR_ZLIB_INITIALIZATION_FAILED",
1211 -1);
1212 } else {
1213 return CompressionError {};
1214 }
1215 }
1216
ResetStream()1217 CompressionError BrotliDecoderContext::ResetStream() {
1218 return Init(alloc_, free_, alloc_opaque_);
1219 }
1220
SetParams(int key,uint32_t value)1221 CompressionError BrotliDecoderContext::SetParams(int key, uint32_t value) {
1222 if (!BrotliDecoderSetParameter(state_.get(),
1223 static_cast<BrotliDecoderParameter>(key),
1224 value)) {
1225 return CompressionError("Setting parameter failed",
1226 "ERR_BROTLI_PARAM_SET_FAILED",
1227 -1);
1228 } else {
1229 return CompressionError {};
1230 }
1231 }
1232
GetErrorInfo() const1233 CompressionError BrotliDecoderContext::GetErrorInfo() const {
1234 if (error_ != BROTLI_DECODER_NO_ERROR) {
1235 return CompressionError("Decompression failed",
1236 error_string_.c_str(),
1237 static_cast<int>(error_));
1238 } else if (flush_ == BROTLI_OPERATION_FINISH &&
1239 last_result_ == BROTLI_DECODER_RESULT_NEEDS_MORE_INPUT) {
1240 // Match zlib's behaviour, as brotli doesn't have its own code for this.
1241 return CompressionError("unexpected end of file",
1242 "Z_BUF_ERROR",
1243 Z_BUF_ERROR);
1244 } else {
1245 return CompressionError {};
1246 }
1247 }
1248
1249
1250 template <typename Stream>
1251 struct MakeClass {
Makenode::__anonb47e24fa0111::MakeClass1252 static void Make(Environment* env, Local<Object> target, const char* name) {
1253 Local<FunctionTemplate> z = env->NewFunctionTemplate(Stream::New);
1254
1255 z->InstanceTemplate()->SetInternalFieldCount(
1256 Stream::kInternalFieldCount);
1257 z->Inherit(AsyncWrap::GetConstructorTemplate(env));
1258
1259 env->SetProtoMethod(z, "write", Stream::template Write<true>);
1260 env->SetProtoMethod(z, "writeSync", Stream::template Write<false>);
1261 env->SetProtoMethod(z, "close", Stream::Close);
1262
1263 env->SetProtoMethod(z, "init", Stream::Init);
1264 env->SetProtoMethod(z, "params", Stream::Params);
1265 env->SetProtoMethod(z, "reset", Stream::Reset);
1266
1267 env->SetConstructorFunction(target, name, z);
1268 }
1269 };
1270
Initialize(Local<Object> target,Local<Value> unused,Local<Context> context,void * priv)1271 void Initialize(Local<Object> target,
1272 Local<Value> unused,
1273 Local<Context> context,
1274 void* priv) {
1275 Environment* env = Environment::GetCurrent(context);
1276
1277 MakeClass<ZlibStream>::Make(env, target, "Zlib");
1278 MakeClass<BrotliEncoderStream>::Make(env, target, "BrotliEncoder");
1279 MakeClass<BrotliDecoderStream>::Make(env, target, "BrotliDecoder");
1280
1281 target->Set(env->context(),
1282 FIXED_ONE_BYTE_STRING(env->isolate(), "ZLIB_VERSION"),
1283 FIXED_ONE_BYTE_STRING(env->isolate(), ZLIB_VERSION)).Check();
1284 }
1285
1286 } // anonymous namespace
1287
DefineZlibConstants(Local<Object> target)1288 void DefineZlibConstants(Local<Object> target) {
1289 NODE_DEFINE_CONSTANT(target, Z_NO_FLUSH);
1290 NODE_DEFINE_CONSTANT(target, Z_PARTIAL_FLUSH);
1291 NODE_DEFINE_CONSTANT(target, Z_SYNC_FLUSH);
1292 NODE_DEFINE_CONSTANT(target, Z_FULL_FLUSH);
1293 NODE_DEFINE_CONSTANT(target, Z_FINISH);
1294 NODE_DEFINE_CONSTANT(target, Z_BLOCK);
1295
1296 // return/error codes
1297 NODE_DEFINE_CONSTANT(target, Z_OK);
1298 NODE_DEFINE_CONSTANT(target, Z_STREAM_END);
1299 NODE_DEFINE_CONSTANT(target, Z_NEED_DICT);
1300 NODE_DEFINE_CONSTANT(target, Z_ERRNO);
1301 NODE_DEFINE_CONSTANT(target, Z_STREAM_ERROR);
1302 NODE_DEFINE_CONSTANT(target, Z_DATA_ERROR);
1303 NODE_DEFINE_CONSTANT(target, Z_MEM_ERROR);
1304 NODE_DEFINE_CONSTANT(target, Z_BUF_ERROR);
1305 NODE_DEFINE_CONSTANT(target, Z_VERSION_ERROR);
1306
1307 NODE_DEFINE_CONSTANT(target, Z_NO_COMPRESSION);
1308 NODE_DEFINE_CONSTANT(target, Z_BEST_SPEED);
1309 NODE_DEFINE_CONSTANT(target, Z_BEST_COMPRESSION);
1310 NODE_DEFINE_CONSTANT(target, Z_DEFAULT_COMPRESSION);
1311 NODE_DEFINE_CONSTANT(target, Z_FILTERED);
1312 NODE_DEFINE_CONSTANT(target, Z_HUFFMAN_ONLY);
1313 NODE_DEFINE_CONSTANT(target, Z_RLE);
1314 NODE_DEFINE_CONSTANT(target, Z_FIXED);
1315 NODE_DEFINE_CONSTANT(target, Z_DEFAULT_STRATEGY);
1316 NODE_DEFINE_CONSTANT(target, ZLIB_VERNUM);
1317
1318 NODE_DEFINE_CONSTANT(target, DEFLATE);
1319 NODE_DEFINE_CONSTANT(target, INFLATE);
1320 NODE_DEFINE_CONSTANT(target, GZIP);
1321 NODE_DEFINE_CONSTANT(target, GUNZIP);
1322 NODE_DEFINE_CONSTANT(target, DEFLATERAW);
1323 NODE_DEFINE_CONSTANT(target, INFLATERAW);
1324 NODE_DEFINE_CONSTANT(target, UNZIP);
1325 NODE_DEFINE_CONSTANT(target, BROTLI_DECODE);
1326 NODE_DEFINE_CONSTANT(target, BROTLI_ENCODE);
1327
1328 NODE_DEFINE_CONSTANT(target, Z_MIN_WINDOWBITS);
1329 NODE_DEFINE_CONSTANT(target, Z_MAX_WINDOWBITS);
1330 NODE_DEFINE_CONSTANT(target, Z_DEFAULT_WINDOWBITS);
1331 NODE_DEFINE_CONSTANT(target, Z_MIN_CHUNK);
1332 NODE_DEFINE_CONSTANT(target, Z_MAX_CHUNK);
1333 NODE_DEFINE_CONSTANT(target, Z_DEFAULT_CHUNK);
1334 NODE_DEFINE_CONSTANT(target, Z_MIN_MEMLEVEL);
1335 NODE_DEFINE_CONSTANT(target, Z_MAX_MEMLEVEL);
1336 NODE_DEFINE_CONSTANT(target, Z_DEFAULT_MEMLEVEL);
1337 NODE_DEFINE_CONSTANT(target, Z_MIN_LEVEL);
1338 NODE_DEFINE_CONSTANT(target, Z_MAX_LEVEL);
1339 NODE_DEFINE_CONSTANT(target, Z_DEFAULT_LEVEL);
1340
1341 // Brotli constants
1342 NODE_DEFINE_CONSTANT(target, BROTLI_OPERATION_PROCESS);
1343 NODE_DEFINE_CONSTANT(target, BROTLI_OPERATION_FLUSH);
1344 NODE_DEFINE_CONSTANT(target, BROTLI_OPERATION_FINISH);
1345 NODE_DEFINE_CONSTANT(target, BROTLI_OPERATION_EMIT_METADATA);
1346 NODE_DEFINE_CONSTANT(target, BROTLI_PARAM_MODE);
1347 NODE_DEFINE_CONSTANT(target, BROTLI_MODE_GENERIC);
1348 NODE_DEFINE_CONSTANT(target, BROTLI_MODE_TEXT);
1349 NODE_DEFINE_CONSTANT(target, BROTLI_MODE_FONT);
1350 NODE_DEFINE_CONSTANT(target, BROTLI_DEFAULT_MODE);
1351 NODE_DEFINE_CONSTANT(target, BROTLI_PARAM_QUALITY);
1352 NODE_DEFINE_CONSTANT(target, BROTLI_MIN_QUALITY);
1353 NODE_DEFINE_CONSTANT(target, BROTLI_MAX_QUALITY);
1354 NODE_DEFINE_CONSTANT(target, BROTLI_DEFAULT_QUALITY);
1355 NODE_DEFINE_CONSTANT(target, BROTLI_PARAM_LGWIN);
1356 NODE_DEFINE_CONSTANT(target, BROTLI_MIN_WINDOW_BITS);
1357 NODE_DEFINE_CONSTANT(target, BROTLI_MAX_WINDOW_BITS);
1358 NODE_DEFINE_CONSTANT(target, BROTLI_LARGE_MAX_WINDOW_BITS);
1359 NODE_DEFINE_CONSTANT(target, BROTLI_DEFAULT_WINDOW);
1360 NODE_DEFINE_CONSTANT(target, BROTLI_PARAM_LGBLOCK);
1361 NODE_DEFINE_CONSTANT(target, BROTLI_MIN_INPUT_BLOCK_BITS);
1362 NODE_DEFINE_CONSTANT(target, BROTLI_MAX_INPUT_BLOCK_BITS);
1363 NODE_DEFINE_CONSTANT(target, BROTLI_PARAM_DISABLE_LITERAL_CONTEXT_MODELING);
1364 NODE_DEFINE_CONSTANT(target, BROTLI_PARAM_SIZE_HINT);
1365 NODE_DEFINE_CONSTANT(target, BROTLI_PARAM_LARGE_WINDOW);
1366 NODE_DEFINE_CONSTANT(target, BROTLI_PARAM_NPOSTFIX);
1367 NODE_DEFINE_CONSTANT(target, BROTLI_PARAM_NDIRECT);
1368 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_RESULT_ERROR);
1369 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_RESULT_SUCCESS);
1370 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_RESULT_NEEDS_MORE_INPUT);
1371 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_RESULT_NEEDS_MORE_OUTPUT);
1372 NODE_DEFINE_CONSTANT(target,
1373 BROTLI_DECODER_PARAM_DISABLE_RING_BUFFER_REALLOCATION);
1374 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_PARAM_LARGE_WINDOW);
1375 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_NO_ERROR);
1376 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_SUCCESS);
1377 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_NEEDS_MORE_INPUT);
1378 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_NEEDS_MORE_OUTPUT);
1379 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_FORMAT_EXUBERANT_NIBBLE);
1380 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_FORMAT_RESERVED);
1381 NODE_DEFINE_CONSTANT(target,
1382 BROTLI_DECODER_ERROR_FORMAT_EXUBERANT_META_NIBBLE);
1383 NODE_DEFINE_CONSTANT(target,
1384 BROTLI_DECODER_ERROR_FORMAT_SIMPLE_HUFFMAN_ALPHABET);
1385 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_FORMAT_SIMPLE_HUFFMAN_SAME);
1386 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_FORMAT_CL_SPACE);
1387 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_FORMAT_HUFFMAN_SPACE);
1388 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_FORMAT_CONTEXT_MAP_REPEAT);
1389 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_FORMAT_BLOCK_LENGTH_1);
1390 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_FORMAT_BLOCK_LENGTH_2);
1391 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_FORMAT_TRANSFORM);
1392 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_FORMAT_DICTIONARY);
1393 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_FORMAT_WINDOW_BITS);
1394 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_FORMAT_PADDING_1);
1395 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_FORMAT_PADDING_2);
1396 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_FORMAT_DISTANCE);
1397 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_DICTIONARY_NOT_SET);
1398 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_INVALID_ARGUMENTS);
1399 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_ALLOC_CONTEXT_MODES);
1400 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_ALLOC_TREE_GROUPS);
1401 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_ALLOC_CONTEXT_MAP);
1402 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_ALLOC_RING_BUFFER_1);
1403 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_ALLOC_RING_BUFFER_2);
1404 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_ALLOC_BLOCK_TYPE_TREES);
1405 NODE_DEFINE_CONSTANT(target, BROTLI_DECODER_ERROR_UNREACHABLE);
1406 }
1407
1408 } // namespace node
1409
1410 NODE_MODULE_CONTEXT_AWARE_INTERNAL(zlib, node::Initialize)
1411