ruby-brs 1.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/AUTHORS +1 -0
- data/LICENSE +21 -0
- data/README.md +339 -0
- data/ext/brs_ext/buffer.c +16 -0
- data/ext/brs_ext/buffer.h +17 -0
- data/ext/brs_ext/common.h +13 -0
- data/ext/brs_ext/error.c +78 -0
- data/ext/brs_ext/error.h +34 -0
- data/ext/brs_ext/io.c +494 -0
- data/ext/brs_ext/io.h +14 -0
- data/ext/brs_ext/macro.h +13 -0
- data/ext/brs_ext/main.c +23 -0
- data/ext/brs_ext/option.c +143 -0
- data/ext/brs_ext/option.h +80 -0
- data/ext/brs_ext/stream/compressor.c +234 -0
- data/ext/brs_ext/stream/compressor.h +31 -0
- data/ext/brs_ext/stream/decompressor.c +181 -0
- data/ext/brs_ext/stream/decompressor.h +29 -0
- data/ext/brs_ext/string.c +274 -0
- data/ext/brs_ext/string.h +14 -0
- data/ext/extconf.rb +66 -0
- data/lib/brs/error.rb +21 -0
- data/lib/brs/file.rb +44 -0
- data/lib/brs/option.rb +80 -0
- data/lib/brs/stream/abstract.rb +153 -0
- data/lib/brs/stream/delegates.rb +36 -0
- data/lib/brs/stream/raw/abstract.rb +55 -0
- data/lib/brs/stream/raw/compressor.rb +97 -0
- data/lib/brs/stream/raw/decompressor.rb +70 -0
- data/lib/brs/stream/reader.rb +166 -0
- data/lib/brs/stream/reader_helpers.rb +192 -0
- data/lib/brs/stream/stat.rb +78 -0
- data/lib/brs/stream/writer.rb +145 -0
- data/lib/brs/stream/writer_helpers.rb +93 -0
- data/lib/brs/string.rb +29 -0
- data/lib/brs/validation.rb +40 -0
- data/lib/brs/version.rb +6 -0
- data/lib/brs.rb +7 -0
- metadata +165 -0
data/ext/brs_ext/io.c
ADDED
@@ -0,0 +1,494 @@
|
|
1
|
+
// Ruby bindings for brotli library.
|
2
|
+
// Copyright (c) 2019 AUTHORS, MIT License.
|
3
|
+
|
4
|
+
#include "ruby/io.h"
|
5
|
+
|
6
|
+
#include <brotli/decode.h>
|
7
|
+
#include <brotli/encode.h>
|
8
|
+
#include <stdint.h>
|
9
|
+
#include <stdio.h>
|
10
|
+
#include <stdlib.h>
|
11
|
+
#include <string.h>
|
12
|
+
|
13
|
+
#include "brs_ext/buffer.h"
|
14
|
+
#include "brs_ext/common.h"
|
15
|
+
#include "brs_ext/error.h"
|
16
|
+
#include "brs_ext/io.h"
|
17
|
+
#include "brs_ext/macro.h"
|
18
|
+
#include "brs_ext/option.h"
|
19
|
+
#include "ruby.h"
|
20
|
+
|
21
|
+
// Additional possible results:
|
22
|
+
enum {
|
23
|
+
BRS_EXT_FILE_READ_FINISHED = 128
|
24
|
+
};
|
25
|
+
|
26
|
+
// -- file --
|
27
|
+
|
28
|
+
static inline brs_ext_result_t read_file(FILE* source_file, uint8_t* source_buffer, size_t* source_length_ptr, size_t source_buffer_length)
|
29
|
+
{
|
30
|
+
size_t read_length = fread(source_buffer, 1, source_buffer_length, source_file);
|
31
|
+
if (read_length == 0 && feof(source_file)) {
|
32
|
+
return BRS_EXT_FILE_READ_FINISHED;
|
33
|
+
}
|
34
|
+
|
35
|
+
if (read_length != source_buffer_length && ferror(source_file)) {
|
36
|
+
return BRS_EXT_ERROR_READ_IO;
|
37
|
+
}
|
38
|
+
|
39
|
+
*source_length_ptr = read_length;
|
40
|
+
|
41
|
+
return 0;
|
42
|
+
}
|
43
|
+
|
44
|
+
static inline brs_ext_result_t write_file(FILE* destination_file, uint8_t* destination_buffer, size_t destination_length)
|
45
|
+
{
|
46
|
+
size_t written_length = fwrite(destination_buffer, 1, destination_length, destination_file);
|
47
|
+
if (written_length != destination_length) {
|
48
|
+
return BRS_EXT_ERROR_WRITE_IO;
|
49
|
+
}
|
50
|
+
|
51
|
+
return 0;
|
52
|
+
}
|
53
|
+
|
54
|
+
// -- buffer --
|
55
|
+
|
56
|
+
static inline brs_ext_result_t create_buffers(
|
57
|
+
uint8_t** source_buffer_ptr, size_t source_buffer_length,
|
58
|
+
uint8_t** destination_buffer_ptr, size_t destination_buffer_length)
|
59
|
+
{
|
60
|
+
uint8_t* source_buffer = malloc(source_buffer_length);
|
61
|
+
if (source_buffer == NULL) {
|
62
|
+
return BRS_EXT_ERROR_ALLOCATE_FAILED;
|
63
|
+
}
|
64
|
+
|
65
|
+
uint8_t* destination_buffer = malloc(destination_buffer_length);
|
66
|
+
if (destination_buffer == NULL) {
|
67
|
+
free(source_buffer);
|
68
|
+
return BRS_EXT_ERROR_ALLOCATE_FAILED;
|
69
|
+
}
|
70
|
+
|
71
|
+
*source_buffer_ptr = source_buffer;
|
72
|
+
*destination_buffer_ptr = destination_buffer;
|
73
|
+
|
74
|
+
return 0;
|
75
|
+
}
|
76
|
+
|
77
|
+
// We have read some source from file into source buffer.
|
78
|
+
// Than algorithm has read part of this source.
|
79
|
+
// We need to move remaining source to the top of source buffer.
|
80
|
+
// Than we can read more source from file.
|
81
|
+
// Algorithm can use same buffer again.
|
82
|
+
|
83
|
+
static inline brs_ext_result_t read_more_source(
|
84
|
+
FILE* source_file,
|
85
|
+
const uint8_t** source_ptr, size_t* source_length_ptr,
|
86
|
+
uint8_t* source_buffer, size_t source_buffer_length)
|
87
|
+
{
|
88
|
+
const uint8_t* source = *source_ptr;
|
89
|
+
size_t source_length = *source_length_ptr;
|
90
|
+
|
91
|
+
if (source != source_buffer) {
|
92
|
+
if (source_length != 0) {
|
93
|
+
memmove(source_buffer, source, source_length);
|
94
|
+
}
|
95
|
+
|
96
|
+
// Source can be accessed even if next code will fail.
|
97
|
+
*source_ptr = source_buffer;
|
98
|
+
}
|
99
|
+
|
100
|
+
size_t remaining_source_buffer_length = source_buffer_length - source_length;
|
101
|
+
if (remaining_source_buffer_length == 0) {
|
102
|
+
// We want to read more data at once, than buffer has.
|
103
|
+
return BRS_EXT_ERROR_NOT_ENOUGH_SOURCE_BUFFER;
|
104
|
+
}
|
105
|
+
|
106
|
+
uint8_t* remaining_source_buffer = source_buffer + source_length;
|
107
|
+
size_t new_source_length;
|
108
|
+
|
109
|
+
brs_ext_result_t ext_result = read_file(source_file, remaining_source_buffer, &new_source_length, remaining_source_buffer_length);
|
110
|
+
if (ext_result != 0) {
|
111
|
+
return ext_result;
|
112
|
+
}
|
113
|
+
|
114
|
+
*source_length_ptr = source_length + new_source_length;
|
115
|
+
|
116
|
+
return 0;
|
117
|
+
}
|
118
|
+
|
119
|
+
#define BUFFERED_READ_SOURCE(function, ...) \
|
120
|
+
while (true) { \
|
121
|
+
ext_result = function(__VA_ARGS__); \
|
122
|
+
if (ext_result != 0) { \
|
123
|
+
return ext_result; \
|
124
|
+
} \
|
125
|
+
\
|
126
|
+
ext_result = read_more_source( \
|
127
|
+
source_file, \
|
128
|
+
&source, &source_length, \
|
129
|
+
source_buffer, source_buffer_length); \
|
130
|
+
\
|
131
|
+
if (ext_result == BRS_EXT_FILE_READ_FINISHED) { \
|
132
|
+
if (source_length != 0) { \
|
133
|
+
/* Brotli wont provide any remainder by design. */ \
|
134
|
+
return BRS_EXT_ERROR_READ_IO; \
|
135
|
+
} \
|
136
|
+
break; \
|
137
|
+
} \
|
138
|
+
else if (ext_result != 0) { \
|
139
|
+
return ext_result; \
|
140
|
+
} \
|
141
|
+
}
|
142
|
+
|
143
|
+
// Algorithm has written data into destination buffer.
|
144
|
+
// We need to write this data into file.
|
145
|
+
// Than algorithm can use same buffer again.
|
146
|
+
|
147
|
+
static inline brs_ext_result_t flush_destination_buffer(
|
148
|
+
FILE* destination_file,
|
149
|
+
uint8_t* destination_buffer, size_t* destination_length_ptr, size_t destination_buffer_length)
|
150
|
+
{
|
151
|
+
if (*destination_length_ptr == 0) {
|
152
|
+
// We want to write more data at once, than buffer has.
|
153
|
+
return BRS_EXT_ERROR_NOT_ENOUGH_DESTINATION_BUFFER;
|
154
|
+
}
|
155
|
+
|
156
|
+
brs_ext_result_t ext_result = write_file(destination_file, destination_buffer, *destination_length_ptr);
|
157
|
+
if (ext_result != 0) {
|
158
|
+
return ext_result;
|
159
|
+
}
|
160
|
+
|
161
|
+
*destination_length_ptr = 0;
|
162
|
+
|
163
|
+
return 0;
|
164
|
+
}
|
165
|
+
|
166
|
+
static inline brs_ext_result_t write_remaining_destination(FILE* destination_file, uint8_t* destination_buffer, size_t destination_length)
|
167
|
+
{
|
168
|
+
if (destination_length == 0) {
|
169
|
+
return 0;
|
170
|
+
}
|
171
|
+
|
172
|
+
return write_file(destination_file, destination_buffer, destination_length);
|
173
|
+
}
|
174
|
+
|
175
|
+
// -- utils --
|
176
|
+
|
177
|
+
#define GET_FILE(target) \
|
178
|
+
Check_Type(target, T_FILE); \
|
179
|
+
\
|
180
|
+
rb_io_t* target##_io; \
|
181
|
+
GetOpenFile(target, target##_io); \
|
182
|
+
\
|
183
|
+
FILE* target##_file = rb_io_stdio_file(target##_io); \
|
184
|
+
if (target##_file == NULL) { \
|
185
|
+
brs_ext_raise_error(BRS_EXT_ERROR_ACCESS_IO); \
|
186
|
+
}
|
187
|
+
|
188
|
+
// -- compress --
|
189
|
+
|
190
|
+
static inline brs_ext_result_t buffered_compress(
|
191
|
+
BrotliEncoderState* state_ptr,
|
192
|
+
const uint8_t** source_ptr, size_t* source_length_ptr,
|
193
|
+
FILE* destination_file, uint8_t* destination_buffer, size_t* destination_length_ptr, size_t destination_buffer_length)
|
194
|
+
{
|
195
|
+
brs_ext_result_t ext_result;
|
196
|
+
|
197
|
+
while (true) {
|
198
|
+
uint8_t* remaining_destination_buffer = destination_buffer + *destination_length_ptr;
|
199
|
+
size_t remaining_destination_buffer_length = destination_buffer_length - *destination_length_ptr;
|
200
|
+
size_t prev_remaining_destination_buffer_length = remaining_destination_buffer_length;
|
201
|
+
|
202
|
+
BROTLI_BOOL result = BrotliEncoderCompressStream(
|
203
|
+
state_ptr,
|
204
|
+
BROTLI_OPERATION_PROCESS,
|
205
|
+
source_length_ptr, source_ptr,
|
206
|
+
&remaining_destination_buffer_length, &remaining_destination_buffer,
|
207
|
+
NULL);
|
208
|
+
|
209
|
+
if (!result) {
|
210
|
+
return BRS_EXT_ERROR_UNEXPECTED;
|
211
|
+
}
|
212
|
+
|
213
|
+
*destination_length_ptr += prev_remaining_destination_buffer_length - remaining_destination_buffer_length;
|
214
|
+
|
215
|
+
if (BrotliEncoderHasMoreOutput(state_ptr)) {
|
216
|
+
ext_result = flush_destination_buffer(
|
217
|
+
destination_file,
|
218
|
+
destination_buffer, destination_length_ptr, destination_buffer_length);
|
219
|
+
|
220
|
+
if (ext_result != 0) {
|
221
|
+
return ext_result;
|
222
|
+
}
|
223
|
+
|
224
|
+
continue;
|
225
|
+
}
|
226
|
+
|
227
|
+
break;
|
228
|
+
}
|
229
|
+
|
230
|
+
return 0;
|
231
|
+
}
|
232
|
+
|
233
|
+
static inline brs_ext_result_t buffered_compressor_finish(
|
234
|
+
BrotliEncoderState* state_ptr,
|
235
|
+
FILE* destination_file, uint8_t* destination_buffer, size_t* destination_length_ptr, size_t destination_buffer_length)
|
236
|
+
{
|
237
|
+
brs_ext_result_t ext_result;
|
238
|
+
|
239
|
+
const uint8_t* source = NULL;
|
240
|
+
size_t source_length = 0;
|
241
|
+
|
242
|
+
while (true) {
|
243
|
+
uint8_t* remaining_destination_buffer = destination_buffer + *destination_length_ptr;
|
244
|
+
size_t remaining_destination_buffer_length = destination_buffer_length - *destination_length_ptr;
|
245
|
+
size_t prev_remaining_destination_buffer_length = remaining_destination_buffer_length;
|
246
|
+
|
247
|
+
BROTLI_BOOL result = BrotliEncoderCompressStream(
|
248
|
+
state_ptr,
|
249
|
+
BROTLI_OPERATION_FINISH,
|
250
|
+
&source_length, &source,
|
251
|
+
&remaining_destination_buffer_length, &remaining_destination_buffer,
|
252
|
+
NULL);
|
253
|
+
|
254
|
+
if (!result) {
|
255
|
+
return BRS_EXT_ERROR_UNEXPECTED;
|
256
|
+
}
|
257
|
+
|
258
|
+
*destination_length_ptr += prev_remaining_destination_buffer_length - remaining_destination_buffer_length;
|
259
|
+
|
260
|
+
if (BrotliEncoderHasMoreOutput(state_ptr) || !BrotliEncoderIsFinished(state_ptr)) {
|
261
|
+
ext_result = flush_destination_buffer(
|
262
|
+
destination_file,
|
263
|
+
destination_buffer, destination_length_ptr, destination_buffer_length);
|
264
|
+
|
265
|
+
if (ext_result != 0) {
|
266
|
+
return ext_result;
|
267
|
+
}
|
268
|
+
|
269
|
+
continue;
|
270
|
+
}
|
271
|
+
|
272
|
+
break;
|
273
|
+
}
|
274
|
+
|
275
|
+
return 0;
|
276
|
+
}
|
277
|
+
|
278
|
+
static inline brs_ext_result_t compress(
|
279
|
+
BrotliEncoderState* state_ptr,
|
280
|
+
FILE* source_file, uint8_t* source_buffer, size_t source_buffer_length,
|
281
|
+
FILE* destination_file, uint8_t* destination_buffer, size_t destination_buffer_length)
|
282
|
+
{
|
283
|
+
brs_ext_result_t ext_result;
|
284
|
+
|
285
|
+
const uint8_t* source = source_buffer;
|
286
|
+
size_t source_length = 0;
|
287
|
+
size_t destination_length = 0;
|
288
|
+
|
289
|
+
BUFFERED_READ_SOURCE(
|
290
|
+
buffered_compress,
|
291
|
+
state_ptr,
|
292
|
+
&source, &source_length,
|
293
|
+
destination_file, destination_buffer, &destination_length, destination_buffer_length);
|
294
|
+
|
295
|
+
ext_result = buffered_compressor_finish(
|
296
|
+
state_ptr,
|
297
|
+
destination_file, destination_buffer, &destination_length, destination_buffer_length);
|
298
|
+
|
299
|
+
if (ext_result != 0) {
|
300
|
+
return ext_result;
|
301
|
+
}
|
302
|
+
|
303
|
+
return write_remaining_destination(destination_file, destination_buffer, destination_length);
|
304
|
+
}
|
305
|
+
|
306
|
+
VALUE brs_ext_compress_io(VALUE BRS_EXT_UNUSED(self), VALUE source, VALUE destination, VALUE options)
|
307
|
+
{
|
308
|
+
GET_FILE(source);
|
309
|
+
GET_FILE(destination);
|
310
|
+
Check_Type(options, T_HASH);
|
311
|
+
BRS_EXT_GET_COMPRESSOR_OPTIONS(options);
|
312
|
+
BRS_EXT_GET_BUFFER_LENGTH_OPTION(options, source_buffer_length);
|
313
|
+
BRS_EXT_GET_BUFFER_LENGTH_OPTION(options, destination_buffer_length);
|
314
|
+
|
315
|
+
BrotliEncoderState* state_ptr = BrotliEncoderCreateInstance(NULL, NULL, NULL);
|
316
|
+
if (state_ptr == NULL) {
|
317
|
+
brs_ext_raise_error(BRS_EXT_ERROR_ALLOCATE_FAILED);
|
318
|
+
}
|
319
|
+
|
320
|
+
brs_ext_result_t ext_result = brs_ext_set_compressor_options(state_ptr, &compressor_options);
|
321
|
+
if (ext_result != 0) {
|
322
|
+
BrotliEncoderDestroyInstance(state_ptr);
|
323
|
+
brs_ext_raise_error(ext_result);
|
324
|
+
}
|
325
|
+
|
326
|
+
if (source_buffer_length == 0) {
|
327
|
+
source_buffer_length = BRS_DEFAULT_SOURCE_BUFFER_LENGTH_FOR_COMPRESSOR;
|
328
|
+
}
|
329
|
+
if (destination_buffer_length == 0) {
|
330
|
+
destination_buffer_length = BRS_DEFAULT_DESTINATION_BUFFER_LENGTH_FOR_COMPRESSOR;
|
331
|
+
}
|
332
|
+
|
333
|
+
uint8_t* source_buffer;
|
334
|
+
uint8_t* destination_buffer;
|
335
|
+
|
336
|
+
ext_result = create_buffers(
|
337
|
+
&source_buffer, source_buffer_length,
|
338
|
+
&destination_buffer, destination_buffer_length);
|
339
|
+
|
340
|
+
if (ext_result != 0) {
|
341
|
+
BrotliEncoderDestroyInstance(state_ptr);
|
342
|
+
brs_ext_raise_error(ext_result);
|
343
|
+
}
|
344
|
+
|
345
|
+
ext_result = compress(
|
346
|
+
state_ptr,
|
347
|
+
source_file, source_buffer, source_buffer_length,
|
348
|
+
destination_file, destination_buffer, destination_buffer_length);
|
349
|
+
|
350
|
+
free(source_buffer);
|
351
|
+
free(destination_buffer);
|
352
|
+
BrotliEncoderDestroyInstance(state_ptr);
|
353
|
+
|
354
|
+
if (ext_result != 0) {
|
355
|
+
brs_ext_raise_error(ext_result);
|
356
|
+
}
|
357
|
+
|
358
|
+
// Ruby itself won't flush stdio file before closing fd, flush is required.
|
359
|
+
fflush(destination_file);
|
360
|
+
|
361
|
+
return Qnil;
|
362
|
+
}
|
363
|
+
|
364
|
+
// -- decompress --
|
365
|
+
|
366
|
+
static inline brs_ext_result_t buffered_decompress(
|
367
|
+
BrotliDecoderState* state_ptr,
|
368
|
+
const uint8_t** source_ptr, size_t* source_length_ptr,
|
369
|
+
FILE* destination_file, uint8_t* destination_buffer, size_t* destination_length_ptr, size_t destination_buffer_length)
|
370
|
+
{
|
371
|
+
brs_ext_result_t ext_result;
|
372
|
+
|
373
|
+
while (true) {
|
374
|
+
uint8_t* remaining_destination_buffer = destination_buffer + *destination_length_ptr;
|
375
|
+
size_t remaining_destination_buffer_length = destination_buffer_length - *destination_length_ptr;
|
376
|
+
size_t prev_remaining_destination_buffer_length = remaining_destination_buffer_length;
|
377
|
+
|
378
|
+
BrotliDecoderResult result = BrotliDecoderDecompressStream(
|
379
|
+
state_ptr,
|
380
|
+
source_length_ptr, source_ptr,
|
381
|
+
&remaining_destination_buffer_length, &remaining_destination_buffer,
|
382
|
+
NULL);
|
383
|
+
|
384
|
+
if (
|
385
|
+
result != BROTLI_DECODER_RESULT_SUCCESS &&
|
386
|
+
result != BROTLI_DECODER_RESULT_NEEDS_MORE_INPUT &&
|
387
|
+
result != BROTLI_DECODER_RESULT_NEEDS_MORE_OUTPUT) {
|
388
|
+
BrotliDecoderErrorCode error_code = BrotliDecoderGetErrorCode(state_ptr);
|
389
|
+
return brs_ext_get_decompressor_error(error_code);
|
390
|
+
}
|
391
|
+
|
392
|
+
*destination_length_ptr += prev_remaining_destination_buffer_length - remaining_destination_buffer_length;
|
393
|
+
|
394
|
+
if (result == BROTLI_DECODER_RESULT_NEEDS_MORE_OUTPUT) {
|
395
|
+
ext_result = flush_destination_buffer(
|
396
|
+
destination_file,
|
397
|
+
destination_buffer, destination_length_ptr, destination_buffer_length);
|
398
|
+
|
399
|
+
if (ext_result != 0) {
|
400
|
+
return ext_result;
|
401
|
+
}
|
402
|
+
|
403
|
+
continue;
|
404
|
+
}
|
405
|
+
|
406
|
+
break;
|
407
|
+
}
|
408
|
+
|
409
|
+
return 0;
|
410
|
+
}
|
411
|
+
|
412
|
+
static inline brs_ext_result_t decompress(
|
413
|
+
BrotliDecoderState* state_ptr,
|
414
|
+
FILE* source_file, uint8_t* source_buffer, size_t source_buffer_length,
|
415
|
+
FILE* destination_file, uint8_t* destination_buffer, size_t destination_buffer_length)
|
416
|
+
{
|
417
|
+
brs_ext_result_t ext_result;
|
418
|
+
|
419
|
+
const uint8_t* source = source_buffer;
|
420
|
+
size_t source_length = 0;
|
421
|
+
size_t destination_length = 0;
|
422
|
+
|
423
|
+
BUFFERED_READ_SOURCE(
|
424
|
+
buffered_decompress,
|
425
|
+
state_ptr,
|
426
|
+
&source, &source_length,
|
427
|
+
destination_file, destination_buffer, &destination_length, destination_buffer_length);
|
428
|
+
|
429
|
+
return write_remaining_destination(destination_file, destination_buffer, destination_length);
|
430
|
+
}
|
431
|
+
|
432
|
+
VALUE brs_ext_decompress_io(VALUE BRS_EXT_UNUSED(self), VALUE source, VALUE destination, VALUE options)
|
433
|
+
{
|
434
|
+
GET_FILE(source);
|
435
|
+
GET_FILE(destination);
|
436
|
+
Check_Type(options, T_HASH);
|
437
|
+
BRS_EXT_GET_DECOMPRESSOR_OPTIONS(options);
|
438
|
+
BRS_EXT_GET_BUFFER_LENGTH_OPTION(options, source_buffer_length);
|
439
|
+
BRS_EXT_GET_BUFFER_LENGTH_OPTION(options, destination_buffer_length);
|
440
|
+
|
441
|
+
BrotliDecoderState* state_ptr = BrotliDecoderCreateInstance(NULL, NULL, NULL);
|
442
|
+
if (state_ptr == NULL) {
|
443
|
+
brs_ext_raise_error(BRS_EXT_ERROR_ALLOCATE_FAILED);
|
444
|
+
}
|
445
|
+
|
446
|
+
brs_ext_result_t ext_result = brs_ext_set_decompressor_options(state_ptr, &decompressor_options);
|
447
|
+
if (ext_result != 0) {
|
448
|
+
BrotliDecoderDestroyInstance(state_ptr);
|
449
|
+
brs_ext_raise_error(ext_result);
|
450
|
+
}
|
451
|
+
|
452
|
+
if (source_buffer_length == 0) {
|
453
|
+
source_buffer_length = BRS_DEFAULT_SOURCE_BUFFER_LENGTH_FOR_DECOMPRESSOR;
|
454
|
+
}
|
455
|
+
if (destination_buffer_length == 0) {
|
456
|
+
destination_buffer_length = BRS_DEFAULT_DESTINATION_BUFFER_LENGTH_FOR_DECOMPRESSOR;
|
457
|
+
}
|
458
|
+
|
459
|
+
uint8_t* source_buffer;
|
460
|
+
uint8_t* destination_buffer;
|
461
|
+
|
462
|
+
ext_result = create_buffers(
|
463
|
+
&source_buffer, source_buffer_length,
|
464
|
+
&destination_buffer, destination_buffer_length);
|
465
|
+
|
466
|
+
if (ext_result != 0) {
|
467
|
+
BrotliDecoderDestroyInstance(state_ptr);
|
468
|
+
brs_ext_raise_error(ext_result);
|
469
|
+
}
|
470
|
+
|
471
|
+
ext_result = decompress(
|
472
|
+
state_ptr,
|
473
|
+
source_file, source_buffer, source_buffer_length,
|
474
|
+
destination_file, destination_buffer, destination_buffer_length);
|
475
|
+
|
476
|
+
free(source_buffer);
|
477
|
+
free(destination_buffer);
|
478
|
+
BrotliDecoderDestroyInstance(state_ptr);
|
479
|
+
|
480
|
+
if (ext_result != 0) {
|
481
|
+
brs_ext_raise_error(ext_result);
|
482
|
+
}
|
483
|
+
|
484
|
+
// Ruby itself won't flush stdio file before closing fd, flush is required.
|
485
|
+
fflush(destination_file);
|
486
|
+
|
487
|
+
return Qnil;
|
488
|
+
}
|
489
|
+
|
490
|
+
void brs_ext_io_exports(VALUE root_module)
|
491
|
+
{
|
492
|
+
rb_define_module_function(root_module, "_native_compress_io", RUBY_METHOD_FUNC(brs_ext_compress_io), 3);
|
493
|
+
rb_define_module_function(root_module, "_native_decompress_io", RUBY_METHOD_FUNC(brs_ext_decompress_io), 3);
|
494
|
+
}
|
data/ext/brs_ext/io.h
ADDED
@@ -0,0 +1,14 @@
|
|
1
|
+
// Ruby bindings for brotli library.
|
2
|
+
// Copyright (c) 2019 AUTHORS, MIT License.
|
3
|
+
|
4
|
+
#if !defined(BRS_EXT_IO_H)
|
5
|
+
#define BRS_EXT_IO_H
|
6
|
+
|
7
|
+
#include "ruby.h"
|
8
|
+
|
9
|
+
VALUE brs_ext_compress_io(VALUE self, VALUE source, VALUE destination, VALUE options);
|
10
|
+
VALUE brs_ext_decompress_io(VALUE self, VALUE source, VALUE destination, VALUE options);
|
11
|
+
|
12
|
+
void brs_ext_io_exports(VALUE root_module);
|
13
|
+
|
14
|
+
#endif // BRS_EXT_IO_H
|
data/ext/brs_ext/macro.h
ADDED
@@ -0,0 +1,13 @@
|
|
1
|
+
// Ruby bindings for brotli library.
|
2
|
+
// Copyright (c) 2019 AUTHORS, MIT License.
|
3
|
+
|
4
|
+
#if !defined(BRS_EXT_MACRO_H)
|
5
|
+
#define BRS_EXT_MACRO_H
|
6
|
+
|
7
|
+
#if defined(__GNUC__)
|
8
|
+
#define BRS_EXT_UNUSED(x) x __attribute__((__unused__))
|
9
|
+
#else
|
10
|
+
#define BRS_EXT_UNUSED(x) x
|
11
|
+
#endif
|
12
|
+
|
13
|
+
#endif // BRS_EXT_MACRO_H
|
data/ext/brs_ext/main.c
ADDED
@@ -0,0 +1,23 @@
|
|
1
|
+
// Ruby bindings for brotli library.
|
2
|
+
// Copyright (c) 2019 AUTHORS, MIT License.
|
3
|
+
|
4
|
+
#include "brs_ext/buffer.h"
|
5
|
+
#include "brs_ext/common.h"
|
6
|
+
#include "brs_ext/io.h"
|
7
|
+
#include "brs_ext/option.h"
|
8
|
+
#include "brs_ext/stream/compressor.h"
|
9
|
+
#include "brs_ext/stream/decompressor.h"
|
10
|
+
#include "brs_ext/string.h"
|
11
|
+
#include "ruby.h"
|
12
|
+
|
13
|
+
void Init_brs_ext()
|
14
|
+
{
|
15
|
+
VALUE root_module = rb_define_module(BRS_EXT_MODULE_NAME);
|
16
|
+
|
17
|
+
brs_ext_buffer_exports(root_module);
|
18
|
+
brs_ext_io_exports(root_module);
|
19
|
+
brs_ext_option_exports(root_module);
|
20
|
+
brs_ext_compressor_exports(root_module);
|
21
|
+
brs_ext_decompressor_exports(root_module);
|
22
|
+
brs_ext_string_exports(root_module);
|
23
|
+
}
|
@@ -0,0 +1,143 @@
|
|
1
|
+
// Ruby bindings for brotli library.
|
2
|
+
// Copyright (c) 2019 AUTHORS, MIT License.
|
3
|
+
|
4
|
+
#include "brs_ext/option.h"
|
5
|
+
|
6
|
+
#include <brotli/decode.h>
|
7
|
+
#include <brotli/encode.h>
|
8
|
+
#include <stdbool.h>
|
9
|
+
#include <stdint.h>
|
10
|
+
|
11
|
+
#include "brs_ext/common.h"
|
12
|
+
#include "brs_ext/error.h"
|
13
|
+
#include "ruby.h"
|
14
|
+
|
15
|
+
static inline VALUE get_raw_option_value(VALUE options, const char* name)
|
16
|
+
{
|
17
|
+
return rb_funcall(options, rb_intern("[]"), 1, ID2SYM(rb_intern(name)));
|
18
|
+
}
|
19
|
+
|
20
|
+
static inline bool get_bool_option_value(VALUE raw_value)
|
21
|
+
{
|
22
|
+
int raw_type = TYPE(raw_value);
|
23
|
+
if (raw_type != T_TRUE && raw_type != T_FALSE) {
|
24
|
+
brs_ext_raise_error(BRS_EXT_ERROR_VALIDATE_FAILED);
|
25
|
+
}
|
26
|
+
|
27
|
+
return raw_type == T_TRUE;
|
28
|
+
}
|
29
|
+
|
30
|
+
static inline unsigned long get_fixnum_option_value(VALUE raw_value)
|
31
|
+
{
|
32
|
+
Check_Type(raw_value, T_FIXNUM);
|
33
|
+
|
34
|
+
return NUM2UINT(raw_value);
|
35
|
+
}
|
36
|
+
|
37
|
+
static inline uint_fast8_t get_mode_option_value(VALUE raw_value)
|
38
|
+
{
|
39
|
+
Check_Type(raw_value, T_SYMBOL);
|
40
|
+
|
41
|
+
ID raw_id = SYM2ID(raw_value);
|
42
|
+
if (raw_id == rb_intern("text")) {
|
43
|
+
return BROTLI_MODE_TEXT;
|
44
|
+
}
|
45
|
+
else if (raw_id == rb_intern("font")) {
|
46
|
+
return BROTLI_MODE_FONT;
|
47
|
+
}
|
48
|
+
else if (raw_id == rb_intern("generic")) {
|
49
|
+
return BROTLI_MODE_GENERIC;
|
50
|
+
}
|
51
|
+
else {
|
52
|
+
brs_ext_raise_error(BRS_EXT_ERROR_VALIDATE_FAILED);
|
53
|
+
}
|
54
|
+
}
|
55
|
+
|
56
|
+
void brs_ext_get_option(VALUE options, brs_ext_option_t* option, brs_ext_option_type_t type, const char* name)
|
57
|
+
{
|
58
|
+
VALUE raw_value = get_raw_option_value(options, name);
|
59
|
+
|
60
|
+
option->has_value = raw_value != Qnil;
|
61
|
+
if (!option->has_value) {
|
62
|
+
return;
|
63
|
+
}
|
64
|
+
|
65
|
+
brs_ext_option_value_t value;
|
66
|
+
|
67
|
+
switch (type) {
|
68
|
+
case BRS_EXT_OPTION_TYPE_BOOL:
|
69
|
+
value = get_bool_option_value(raw_value) ? 1 : 0;
|
70
|
+
break;
|
71
|
+
case BRS_EXT_OPTION_TYPE_FIXNUM:
|
72
|
+
value = (brs_ext_option_value_t)get_fixnum_option_value(raw_value);
|
73
|
+
break;
|
74
|
+
case BRS_EXT_OPTION_TYPE_MODE:
|
75
|
+
value = (brs_ext_option_value_t)get_mode_option_value(raw_value);
|
76
|
+
break;
|
77
|
+
default:
|
78
|
+
brs_ext_raise_error(BRS_EXT_ERROR_UNEXPECTED);
|
79
|
+
}
|
80
|
+
|
81
|
+
option->value = value;
|
82
|
+
}
|
83
|
+
|
84
|
+
unsigned long brs_ext_get_fixnum_option_value(VALUE options, const char* name)
|
85
|
+
{
|
86
|
+
VALUE raw_value = get_raw_option_value(options, name);
|
87
|
+
|
88
|
+
return get_fixnum_option_value(raw_value);
|
89
|
+
}
|
90
|
+
|
91
|
+
#define SET_OPTION_VALUE(function, state_ptr, param, option) \
|
92
|
+
if (option.has_value && !function(state_ptr, param, option.value)) { \
|
93
|
+
return BRS_EXT_ERROR_VALIDATE_FAILED; \
|
94
|
+
}
|
95
|
+
|
96
|
+
#define SET_ENCODER_PARAM(state_ptr, param, option) \
|
97
|
+
SET_OPTION_VALUE(BrotliEncoderSetParameter, state_ptr, param, option);
|
98
|
+
|
99
|
+
brs_ext_result_t brs_ext_set_compressor_options(BrotliEncoderState* state_ptr, brs_ext_compressor_options_t* options)
|
100
|
+
{
|
101
|
+
SET_ENCODER_PARAM(state_ptr, BROTLI_PARAM_MODE, options->mode);
|
102
|
+
SET_ENCODER_PARAM(state_ptr, BROTLI_PARAM_QUALITY, options->quality);
|
103
|
+
SET_ENCODER_PARAM(state_ptr, BROTLI_PARAM_LGWIN, options->lgwin);
|
104
|
+
SET_ENCODER_PARAM(state_ptr, BROTLI_PARAM_LGBLOCK, options->lgblock);
|
105
|
+
SET_ENCODER_PARAM(state_ptr, BROTLI_PARAM_DISABLE_LITERAL_CONTEXT_MODELING, options->disable_literal_context_modeling);
|
106
|
+
SET_ENCODER_PARAM(state_ptr, BROTLI_PARAM_SIZE_HINT, options->size_hint);
|
107
|
+
SET_ENCODER_PARAM(state_ptr, BROTLI_PARAM_LARGE_WINDOW, options->large_window);
|
108
|
+
|
109
|
+
return 0;
|
110
|
+
}
|
111
|
+
|
112
|
+
#define SET_DECODER_PARAM(state_ptr, param, option) \
|
113
|
+
SET_OPTION_VALUE(BrotliDecoderSetParameter, state_ptr, param, option);
|
114
|
+
|
115
|
+
brs_ext_result_t brs_ext_set_decompressor_options(BrotliDecoderState* state_ptr, brs_ext_decompressor_options_t* options)
|
116
|
+
{
|
117
|
+
SET_DECODER_PARAM(state_ptr, BROTLI_DECODER_PARAM_DISABLE_RING_BUFFER_REALLOCATION, options->disable_ring_buffer_reallocation);
|
118
|
+
SET_DECODER_PARAM(state_ptr, BROTLI_DECODER_PARAM_LARGE_WINDOW, options->large_window);
|
119
|
+
|
120
|
+
return 0;
|
121
|
+
}
|
122
|
+
|
123
|
+
void brs_ext_option_exports(VALUE root_module)
|
124
|
+
{
|
125
|
+
VALUE option = rb_define_module_under(root_module, "Option");
|
126
|
+
|
127
|
+
VALUE modes = rb_ary_new_from_args(
|
128
|
+
3,
|
129
|
+
ID2SYM(rb_intern("text")),
|
130
|
+
ID2SYM(rb_intern("font")),
|
131
|
+
ID2SYM(rb_intern("generic")));
|
132
|
+
rb_define_const(option, "MODES", modes);
|
133
|
+
RB_GC_GUARD(modes);
|
134
|
+
|
135
|
+
rb_define_const(option, "MIN_QUALITY", UINT2NUM(BROTLI_MIN_QUALITY));
|
136
|
+
rb_define_const(option, "MAX_QUALITY", UINT2NUM(BROTLI_MAX_QUALITY));
|
137
|
+
|
138
|
+
rb_define_const(option, "MIN_LGWIN", UINT2NUM(BROTLI_MIN_WINDOW_BITS));
|
139
|
+
rb_define_const(option, "MAX_LGWIN", UINT2NUM(BROTLI_MAX_WINDOW_BITS));
|
140
|
+
|
141
|
+
rb_define_const(option, "MIN_LGBLOCK", UINT2NUM(BROTLI_MIN_INPUT_BLOCK_BITS));
|
142
|
+
rb_define_const(option, "MAX_LGBLOCK", UINT2NUM(BROTLI_MAX_INPUT_BLOCK_BITS));
|
143
|
+
}
|