16#include "internal/array.h"
17#include "internal/bits.h"
18#include "internal/error.h"
19#include "internal/gc.h"
20#include "internal/numeric.h"
21#include "internal/string.h"
22#include "internal/io.h"
25VALUE rb_eIOBufferLockedError;
26VALUE rb_eIOBufferAllocationError;
27VALUE rb_eIOBufferAccessError;
28VALUE rb_eIOBufferInvalidatedError;
29VALUE rb_eIOBufferMaskError;
31size_t RUBY_IO_BUFFER_PAGE_SIZE;
32size_t RUBY_IO_BUFFER_DEFAULT_SIZE;
41 RB_IO_BUFFER_HEXDUMP_DEFAULT_WIDTH = 16,
42 RB_IO_BUFFER_HEXDUMP_MAXIMUM_WIDTH = 1024,
44 RB_IO_BUFFER_INSPECT_HEXDUMP_MAXIMUM_SIZE = 256,
45 RB_IO_BUFFER_INSPECT_HEXDUMP_WIDTH = 16,
48 RB_IO_BUFFER_FLAGS_MASK = RB_IO_BUFFER_EXTERNAL | RB_IO_BUFFER_INTERNAL | RB_IO_BUFFER_MAPPED | RB_IO_BUFFER_SHARED | RB_IO_BUFFER_LOCKED | RB_IO_BUFFER_PRIVATE | RB_IO_BUFFER_READONLY,
50 RB_IO_BUFFER_DEBUG = 0,
56 enum rb_io_buffer_flags flags;
66io_buffer_map_memory(
size_t size,
int flags)
69 void * base = VirtualAlloc(0, size, MEM_COMMIT, PAGE_READWRITE);
72 rb_sys_fail(
"io_buffer_map_memory:VirtualAlloc");
75 int mmap_flags = MAP_ANONYMOUS;
76 if (flags & RB_IO_BUFFER_SHARED) {
77 mmap_flags |= MAP_SHARED;
80 mmap_flags |= MAP_PRIVATE;
83 void * base = mmap(NULL, size, PROT_READ | PROT_WRITE, mmap_flags, -1, 0);
85 if (base == MAP_FAILED) {
86 rb_sys_fail(
"io_buffer_map_memory:mmap");
89 ruby_annotate_mmap(base, size,
"Ruby:io_buffer_map_memory");
96io_buffer_map_file(
struct rb_io_buffer *buffer,
int descriptor,
size_t size, rb_off_t offset,
enum rb_io_buffer_flags flags)
99 HANDLE file = (HANDLE)_get_osfhandle(descriptor);
100 if (!file) rb_sys_fail(
"io_buffer_map_descriptor:_get_osfhandle");
102 DWORD protect = PAGE_READONLY, access = FILE_MAP_READ;
104 if (flags & RB_IO_BUFFER_READONLY) {
105 buffer->flags |= RB_IO_BUFFER_READONLY;
108 protect = PAGE_READWRITE;
109 access = FILE_MAP_WRITE;
112 if (flags & RB_IO_BUFFER_PRIVATE) {
113 protect = PAGE_WRITECOPY;
114 access = FILE_MAP_COPY;
115 buffer->flags |= RB_IO_BUFFER_PRIVATE;
119 buffer->flags |= RB_IO_BUFFER_EXTERNAL;
120 buffer->flags |= RB_IO_BUFFER_SHARED;
123 HANDLE mapping = CreateFileMapping(file, NULL, protect, 0, 0, NULL);
124 if (RB_IO_BUFFER_DEBUG) fprintf(stderr,
"io_buffer_map_file:CreateFileMapping -> %p\n", mapping);
125 if (!mapping) rb_sys_fail(
"io_buffer_map_descriptor:CreateFileMapping");
127 void *base = MapViewOfFile(mapping, access, (DWORD)(offset >> 32), (DWORD)(offset & 0xFFFFFFFF), size);
130 CloseHandle(mapping);
131 rb_sys_fail(
"io_buffer_map_file:MapViewOfFile");
134 buffer->mapping = mapping;
136 int protect = PROT_READ, access = 0;
138 if (flags & RB_IO_BUFFER_READONLY) {
139 buffer->flags |= RB_IO_BUFFER_READONLY;
142 protect |= PROT_WRITE;
145 if (flags & RB_IO_BUFFER_PRIVATE) {
146 buffer->flags |= RB_IO_BUFFER_PRIVATE;
147 access |= MAP_PRIVATE;
151 buffer->flags |= RB_IO_BUFFER_EXTERNAL;
152 buffer->flags |= RB_IO_BUFFER_SHARED;
153 access |= MAP_SHARED;
156 void *base = mmap(NULL, size, protect, access, descriptor, offset);
158 if (base == MAP_FAILED) {
159 rb_sys_fail(
"io_buffer_map_file:mmap");
166 buffer->flags |= RB_IO_BUFFER_MAPPED;
167 buffer->flags |= RB_IO_BUFFER_FILE;
171io_buffer_experimental(
void)
173 static int warned = 0;
181 "IO::Buffer is experimental and both the Ruby and C interface may change in the future!"
192 buffer->mapping = NULL;
194 buffer->source =
Qnil;
198io_buffer_initialize(
VALUE self,
struct rb_io_buffer *buffer,
void *base,
size_t size,
enum rb_io_buffer_flags flags,
VALUE source)
205 if (flags & RB_IO_BUFFER_INTERNAL) {
206 base = calloc(size, 1);
208 else if (flags & RB_IO_BUFFER_MAPPED) {
209 base = io_buffer_map_memory(size, flags);
213 rb_raise(rb_eIOBufferAllocationError,
"Could not allocate buffer!");
223 buffer->flags = flags;
227 buffer->mapping = NULL;
235 if (buffer->flags & RB_IO_BUFFER_INTERNAL) {
239 if (buffer->flags & RB_IO_BUFFER_MAPPED) {
241 if (buffer->flags & RB_IO_BUFFER_FILE) {
242 UnmapViewOfFile(buffer->base);
245 VirtualFree(buffer->base, 0, MEM_RELEASE);
248 munmap(buffer->base, buffer->size);
262 buffer->source =
Qnil;
266 if (buffer->mapping) {
267 if (RB_IO_BUFFER_DEBUG) fprintf(stderr,
"io_buffer_free:CloseHandle -> %p\n", buffer->mapping);
268 if (!CloseHandle(buffer->mapping)) {
269 fprintf(stderr,
"io_buffer_free:GetLastError -> %lu\n", GetLastError());
271 buffer->mapping = NULL;
277rb_io_buffer_type_mark(
void *_buffer)
280 if (buffer->source !=
Qnil) {
284 rb_gc_mark(buffer->source);
286 rb_gc_mark_movable(buffer->source);
292rb_io_buffer_type_compact(
void *_buffer)
295 if (buffer->source !=
Qnil) {
300 buffer->source = rb_gc_location(buffer->source);
306rb_io_buffer_type_free(
void *_buffer)
310 io_buffer_free(buffer);
314rb_io_buffer_type_size(
const void *_buffer)
320 total += buffer->size;
329 .dmark = rb_io_buffer_type_mark,
330 .dfree = rb_io_buffer_type_free,
331 .dsize = rb_io_buffer_type_size,
332 .dcompact = rb_io_buffer_type_compact,
338static inline enum rb_io_buffer_flags
339io_buffer_extract_flags(
VALUE argument)
341 if (rb_int_negative_p(argument)) {
342 rb_raise(rb_eArgError,
"Flags can't be negative!");
345 enum rb_io_buffer_flags flags =
RB_NUM2UINT(argument);
348 return flags & RB_IO_BUFFER_FLAGS_MASK;
353io_buffer_extract_offset(
VALUE argument)
355 if (rb_int_negative_p(argument)) {
356 rb_raise(rb_eArgError,
"Offset can't be negative!");
366io_buffer_extract_length(
VALUE argument)
368 if (rb_int_negative_p(argument)) {
369 rb_raise(rb_eArgError,
"Length can't be negative!");
378io_buffer_extract_size(
VALUE argument)
380 if (rb_int_negative_p(argument)) {
381 rb_raise(rb_eArgError,
"Size can't be negative!");
390io_buffer_extract_width(
VALUE argument,
size_t minimum)
392 if (rb_int_negative_p(argument)) {
393 rb_raise(rb_eArgError,
"Width can't be negative!");
398 if (width < minimum) {
399 rb_raise(rb_eArgError,
"Width must be at least %" PRIuSIZE
"!", minimum);
402 if (width > RB_IO_BUFFER_HEXDUMP_MAXIMUM_WIDTH) {
403 rb_raise(rb_eArgError,
"Width must be at most %" PRIuSIZE
"!", (
size_t)RB_IO_BUFFER_HEXDUMP_MAXIMUM_WIDTH);
414io_buffer_default_length(
const struct rb_io_buffer *buffer,
size_t offset)
416 if (offset > buffer->size) {
417 rb_raise(rb_eArgError,
"The given offset is bigger than the buffer size!");
421 return buffer->size - offset;
430io_buffer_extract_length_offset(
VALUE self,
int argc,
VALUE argv[],
size_t *length,
size_t *offset)
435 if (argc >= 2 && !
NIL_P(argv[1])) {
436 *offset = io_buffer_extract_offset(argv[1]);
442 if (argc >= 1 && !
NIL_P(argv[0])) {
443 *length = io_buffer_extract_length(argv[0]);
446 *length = io_buffer_default_length(buffer, *offset);
462io_buffer_extract_offset_length(
VALUE self,
int argc,
VALUE argv[],
size_t *offset,
size_t *length)
467 if (argc >= 1 && !
NIL_P(argv[0])) {
468 *offset = io_buffer_extract_offset(argv[0]);
474 if (argc >= 2 && !
NIL_P(argv[1])) {
475 *length = io_buffer_extract_length(argv[1]);
478 *length = io_buffer_default_length(buffer, *offset);
485rb_io_buffer_type_allocate(
VALUE self)
487 io_buffer_experimental();
492 io_buffer_zero(buffer);
497static VALUE io_buffer_for_make_instance(
VALUE klass,
VALUE string,
enum rb_io_buffer_flags flags)
499 VALUE instance = rb_io_buffer_type_allocate(klass);
504 flags |= RB_IO_BUFFER_EXTERNAL;
507 flags |= RB_IO_BUFFER_READONLY;
509 if (!(flags & RB_IO_BUFFER_READONLY))
510 rb_str_modify(
string);
512 io_buffer_initialize(instance, buffer, RSTRING_PTR(
string), RSTRING_LEN(
string), flags,
string);
521 enum rb_io_buffer_flags flags;
525io_buffer_for_yield_instance(
VALUE _arguments)
529 arguments->instance = io_buffer_for_make_instance(arguments->klass, arguments->string, arguments->flags);
535 return rb_yield(arguments->instance);
539io_buffer_for_yield_instance_ensure(
VALUE _arguments)
543 if (arguments->instance !=
Qnil) {
544 rb_io_buffer_free(arguments->instance);
591rb_io_buffer_type_for(
VALUE klass,
VALUE string)
605 return rb_ensure(io_buffer_for_yield_instance, (
VALUE)&arguments, io_buffer_for_yield_instance_ensure, (
VALUE)&arguments);
609 string = rb_str_tmp_frozen_acquire(
string);
610 return io_buffer_for_make_instance(klass,
string, RB_IO_BUFFER_READONLY);
628rb_io_buffer_type_string(
VALUE klass,
VALUE length)
638 rb_ensure(io_buffer_for_yield_instance, (
VALUE)&arguments, io_buffer_for_yield_instance_ensure, (
VALUE)&arguments);
644rb_io_buffer_new(
void *base,
size_t size,
enum rb_io_buffer_flags flags)
646 VALUE instance = rb_io_buffer_type_allocate(rb_cIOBuffer);
651 io_buffer_initialize(instance, buffer, base, size, flags,
Qnil);
657rb_io_buffer_map(
VALUE io,
size_t size, rb_off_t offset,
enum rb_io_buffer_flags flags)
659 VALUE instance = rb_io_buffer_type_allocate(rb_cIOBuffer);
666 io_buffer_map_file(buffer, descriptor, size, offset, flags);
716io_buffer_map(
int argc,
VALUE *argv,
VALUE klass)
723 rb_off_t file_size = rb_file_size(io);
725 if (UNLIKELY(file_size <= 0)) {
726 rb_raise(rb_eArgError,
"Invalid negative or zero file size!");
729 else if (UNLIKELY((uintmax_t)file_size > SIZE_MAX)) {
730 rb_raise(rb_eArgError,
"File larger than address space!");
734 if (argc >= 2 && !
RB_NIL_P(argv[1])) {
735 size = io_buffer_extract_size(argv[1]);
736 if (UNLIKELY(size == 0)) {
737 rb_raise(rb_eArgError,
"Size can't be zero!");
739 if (UNLIKELY(size > (
size_t)file_size)) {
740 rb_raise(rb_eArgError,
"Size can't be larger than file size!");
745 size = (size_t)file_size;
752 if (UNLIKELY(offset < 0)) {
753 rb_raise(rb_eArgError,
"Offset can't be negative!");
755 if (UNLIKELY(offset >= file_size)) {
756 rb_raise(rb_eArgError,
"Offset too large!");
760 size = (size_t)(file_size - offset);
762 else if (UNLIKELY((
size_t)(file_size - offset) < size)) {
763 rb_raise(rb_eArgError,
"Offset too large!");
767 enum rb_io_buffer_flags flags = 0;
769 flags = io_buffer_extract_flags(argv[3]);
772 return rb_io_buffer_map(io, size, offset, flags);
776static inline enum rb_io_buffer_flags
777io_flags_for_size(
size_t size)
779 if (size >= RUBY_IO_BUFFER_PAGE_SIZE) {
780 return RB_IO_BUFFER_MAPPED;
783 return RB_IO_BUFFER_INTERNAL;
811rb_io_buffer_initialize(
int argc,
VALUE *argv,
VALUE self)
820 size = io_buffer_extract_size(argv[0]);
823 size = RUBY_IO_BUFFER_DEFAULT_SIZE;
826 enum rb_io_buffer_flags flags = 0;
828 flags = io_buffer_extract_flags(argv[1]);
831 flags |= io_flags_for_size(size);
834 io_buffer_initialize(self, buffer, NULL, size, flags,
Qnil);
840io_buffer_validate_slice(
VALUE source,
void *base,
size_t size)
842 void *source_base = NULL;
843 size_t source_size = 0;
849 rb_io_buffer_get_bytes(source, &source_base, &source_size);
853 if (source_base == NULL)
return 0;
856 if (base < source_base)
return 0;
858 const void *source_end = (
char*)source_base + source_size;
859 const void *end = (
char*)base + size;
862 if (end > source_end)
return 0;
871 if (buffer->source !=
Qnil) {
873 return io_buffer_validate_slice(buffer->source, buffer->base, buffer->size);
880enum rb_io_buffer_flags
881rb_io_buffer_get_bytes(
VALUE self,
void **base,
size_t *size)
886 if (io_buffer_validate(buffer)) {
888 *base = buffer->base;
889 *size = buffer->size;
891 return buffer->flags;
903io_buffer_get_bytes_for_writing(
struct rb_io_buffer *buffer,
void **base,
size_t *size)
905 if (buffer->flags & RB_IO_BUFFER_READONLY ||
907 rb_raise(rb_eIOBufferAccessError,
"Buffer is not writable!");
910 if (!io_buffer_validate(buffer)) {
911 rb_raise(rb_eIOBufferInvalidatedError,
"Buffer is invalid!");
915 *base = buffer->base;
916 *size = buffer->size;
924rb_io_buffer_get_bytes_for_writing(
VALUE self,
void **base,
size_t *size)
929 io_buffer_get_bytes_for_writing(buffer, base, size);
933io_buffer_get_bytes_for_reading(
struct rb_io_buffer *buffer,
const void **base,
size_t *size)
935 if (!io_buffer_validate(buffer)) {
936 rb_raise(rb_eIOBufferInvalidatedError,
"Buffer has been invalidated!");
940 *base = buffer->base;
941 *size = buffer->size;
949rb_io_buffer_get_bytes_for_reading(
VALUE self,
const void **base,
size_t *size)
954 io_buffer_get_bytes_for_reading(buffer, base, size);
967rb_io_buffer_to_s(
VALUE self)
975 rb_str_catf(result,
" %p+%"PRIdSIZE, buffer->base, buffer->size);
977 if (buffer->base == NULL) {
981 if (buffer->flags & RB_IO_BUFFER_EXTERNAL) {
985 if (buffer->flags & RB_IO_BUFFER_INTERNAL) {
989 if (buffer->flags & RB_IO_BUFFER_MAPPED) {
993 if (buffer->flags & RB_IO_BUFFER_FILE) {
997 if (buffer->flags & RB_IO_BUFFER_SHARED) {
1001 if (buffer->flags & RB_IO_BUFFER_LOCKED) {
1005 if (buffer->flags & RB_IO_BUFFER_PRIVATE) {
1009 if (buffer->flags & RB_IO_BUFFER_READONLY) {
1013 if (buffer->source !=
Qnil) {
1017 if (!io_buffer_validate(buffer)) {
1027io_buffer_hexdump_output_size(
size_t width,
size_t size,
int first)
1030 size_t total = size;
1032 size_t whole_lines = (size / width);
1033 size_t partial_line = (size % width) ? 1 : 0;
1038 total += (whole_lines + partial_line) * (1 + 10 + width*3 + 1 + 1);
1041 if (size && first) total -= 1;
1050io_buffer_hexdump(
VALUE string,
size_t width,
const char *base,
size_t length,
size_t offset,
int first)
1052 char *text = alloca(width+1);
1055 for (; offset < length; offset += width) {
1056 memset(text,
'\0', width);
1058 rb_str_catf(
string,
"0x%08" PRIxSIZE
" ", offset);
1062 rb_str_catf(
string,
"\n0x%08" PRIxSIZE
" ", offset);
1065 for (
size_t i = 0; i < width; i += 1) {
1066 if (offset+i < length) {
1067 unsigned char value = ((
unsigned char*)base)[offset+i];
1069 if (value < 127 && isprint(value)) {
1070 text[i] = (char)value;
1076 rb_str_catf(
string,
" %02x", value);
1083 rb_str_catf(
string,
" %s", text);
1102rb_io_buffer_inspect(
VALUE self)
1107 VALUE result = rb_io_buffer_to_s(self);
1109 if (io_buffer_validate(buffer)) {
1111 size_t size = buffer->size;
1114 if (size > RB_IO_BUFFER_INSPECT_HEXDUMP_MAXIMUM_SIZE) {
1115 size = RB_IO_BUFFER_INSPECT_HEXDUMP_MAXIMUM_SIZE;
1119 io_buffer_hexdump(result, RB_IO_BUFFER_INSPECT_HEXDUMP_WIDTH, buffer->base, size, 0, 0);
1122 rb_str_catf(result,
"\n(and %" PRIuSIZE
" more bytes not printed)", buffer->size - size);
1136rb_io_buffer_size(
VALUE self)
1153rb_io_buffer_valid_p(
VALUE self)
1158 return RBOOL(io_buffer_validate(buffer));
1176rb_io_buffer_null_p(
VALUE self)
1181 return RBOOL(buffer->base == NULL);
1192rb_io_buffer_empty_p(
VALUE self)
1197 return RBOOL(buffer->size == 0);
1212rb_io_buffer_external_p(
VALUE self)
1217 return RBOOL(buffer->flags & RB_IO_BUFFER_EXTERNAL);
1237rb_io_buffer_internal_p(
VALUE self)
1242 return RBOOL(buffer->flags & RB_IO_BUFFER_INTERNAL);
1259rb_io_buffer_mapped_p(
VALUE self)
1264 return RBOOL(buffer->flags & RB_IO_BUFFER_MAPPED);
1291rb_io_buffer_shared_p(
VALUE self)
1296 return RBOOL(buffer->flags & RB_IO_BUFFER_SHARED);
1314rb_io_buffer_locked_p(
VALUE self)
1319 return RBOOL(buffer->flags & RB_IO_BUFFER_LOCKED);
1345rb_io_buffer_private_p(
VALUE self)
1350 return RBOOL(buffer->flags & RB_IO_BUFFER_PRIVATE);
1354rb_io_buffer_readonly_p(
VALUE self)
1359 return buffer->flags & RB_IO_BUFFER_READONLY;
1371io_buffer_readonly_p(
VALUE self)
1373 return RBOOL(rb_io_buffer_readonly_p(self));
1379 if (buffer->flags & RB_IO_BUFFER_LOCKED) {
1380 rb_raise(rb_eIOBufferLockedError,
"Buffer already locked!");
1383 buffer->flags |= RB_IO_BUFFER_LOCKED;
1387rb_io_buffer_lock(
VALUE self)
1392 io_buffer_lock(buffer);
1400 if (!(buffer->flags & RB_IO_BUFFER_LOCKED)) {
1401 rb_raise(rb_eIOBufferLockedError,
"Buffer not locked!");
1404 buffer->flags &= ~RB_IO_BUFFER_LOCKED;
1408rb_io_buffer_unlock(
VALUE self)
1413 io_buffer_unlock(buffer);
1419rb_io_buffer_try_unlock(
VALUE self)
1424 if (buffer->flags & RB_IO_BUFFER_LOCKED) {
1425 buffer->flags &= ~RB_IO_BUFFER_LOCKED;
1433rb_io_buffer_locked_ensure(
VALUE self)
1438 buffer->flags &= ~RB_IO_BUFFER_LOCKED;
1474rb_io_buffer_locked(
VALUE self)
1479 if (buffer->flags & RB_IO_BUFFER_LOCKED) {
1480 rb_raise(rb_eIOBufferLockedError,
"Buffer already locked!");
1483 buffer->flags |= RB_IO_BUFFER_LOCKED;
1514rb_io_buffer_free(
VALUE self)
1519 if (buffer->flags & RB_IO_BUFFER_LOCKED) {
1520 rb_raise(rb_eIOBufferLockedError,
"Buffer is locked!");
1523 io_buffer_free(buffer);
1533 io_buffer_unlock(buffer);
1534 io_buffer_free(buffer);
1540size_sum_is_bigger_than(
size_t a,
size_t b,
size_t x)
1542 struct rbimpl_size_overflow_tag size = rbimpl_size_add_overflow(a, b);
1543 return size.overflowed || size.result > x;
1549io_buffer_validate_range(
struct rb_io_buffer *buffer,
size_t offset,
size_t length)
1551 if (size_sum_is_bigger_than(offset, length, buffer->size)) {
1552 rb_raise(rb_eArgError,
"Specified offset+length is bigger than the buffer size!");
1573rb_io_buffer_hexdump(
int argc,
VALUE *argv,
VALUE self)
1577 size_t offset, length;
1578 struct rb_io_buffer *buffer = io_buffer_extract_offset_length(self, argc, argv, &offset, &length);
1580 size_t width = RB_IO_BUFFER_HEXDUMP_DEFAULT_WIDTH;
1582 width = io_buffer_extract_width(argv[2], 1);
1586 io_buffer_validate_range(buffer, offset, length);
1590 if (io_buffer_validate(buffer) && buffer->base) {
1591 result =
rb_str_buf_new(io_buffer_hexdump_output_size(width, length, 1));
1593 io_buffer_hexdump(result, width, buffer->base, offset+length, offset, 1);
1600rb_io_buffer_slice(
struct rb_io_buffer *buffer,
VALUE self,
size_t offset,
size_t length)
1602 io_buffer_validate_range(buffer, offset, length);
1608 slice->flags |= (buffer->flags & RB_IO_BUFFER_READONLY);
1609 slice->base = (
char*)buffer->base + offset;
1610 slice->size = length;
1613 if (buffer->source !=
Qnil) {
1614 RB_OBJ_WRITE(instance, &slice->source, buffer->source);
1674io_buffer_slice(
int argc,
VALUE *argv,
VALUE self)
1678 size_t offset, length;
1679 struct rb_io_buffer *buffer = io_buffer_extract_offset_length(self, argc, argv, &offset, &length);
1681 return rb_io_buffer_slice(buffer, self, offset, length);
1703rb_io_buffer_transfer(
VALUE self)
1708 if (buffer->flags & RB_IO_BUFFER_LOCKED) {
1709 rb_raise(rb_eIOBufferLockedError,
"Cannot transfer ownership of locked buffer!");
1716 *transferred = *buffer;
1717 io_buffer_zero(buffer);
1723io_buffer_resize_clear(
struct rb_io_buffer *buffer,
void* base,
size_t size)
1725 if (size > buffer->size) {
1726 memset((
unsigned char*)base+buffer->size, 0, size - buffer->size);
1735 io_buffer_initialize(self, &resized, NULL, size, io_flags_for_size(size),
Qnil);
1738 size_t preserve = buffer->size;
1739 if (preserve > size) preserve = size;
1740 memcpy(resized.base, buffer->base, preserve);
1742 io_buffer_resize_clear(buffer, resized.base, size);
1745 io_buffer_free(buffer);
1750rb_io_buffer_resize(
VALUE self,
size_t size)
1755 if (buffer->flags & RB_IO_BUFFER_LOCKED) {
1756 rb_raise(rb_eIOBufferLockedError,
"Cannot resize locked buffer!");
1759 if (buffer->base == NULL) {
1760 io_buffer_initialize(self, buffer, NULL, size, io_flags_for_size(size),
Qnil);
1764 if (buffer->flags & RB_IO_BUFFER_EXTERNAL) {
1765 rb_raise(rb_eIOBufferAccessError,
"Cannot resize external buffer!");
1768#if defined(HAVE_MREMAP) && defined(MREMAP_MAYMOVE)
1769 if (buffer->flags & RB_IO_BUFFER_MAPPED) {
1770 void *base = mremap(buffer->base, buffer->size, size, MREMAP_MAYMOVE);
1772 if (base == MAP_FAILED) {
1773 rb_sys_fail(
"rb_io_buffer_resize:mremap");
1776 io_buffer_resize_clear(buffer, base, size);
1778 buffer->base = base;
1779 buffer->size = size;
1785 if (buffer->flags & RB_IO_BUFFER_INTERNAL) {
1787 io_buffer_free(buffer);
1791 void *base = realloc(buffer->base, size);
1794 rb_sys_fail(
"rb_io_buffer_resize:realloc");
1797 io_buffer_resize_clear(buffer, base, size);
1799 buffer->base = base;
1800 buffer->size = size;
1805 io_buffer_resize_copy(self, buffer, size);
1829 rb_io_buffer_resize(self, io_buffer_extract_size(size));
1843 const void *ptr1, *ptr2;
1844 size_t size1, size2;
1846 rb_io_buffer_get_bytes_for_reading(self, &ptr1, &size1);
1847 rb_io_buffer_get_bytes_for_reading(other, &ptr2, &size2);
1849 if (size1 < size2) {
1853 if (size1 > size2) {
1857 return RB_INT2NUM(memcmp(ptr1, ptr2, size1));
1861io_buffer_validate_type(
size_t size,
size_t offset,
size_t extend)
1863 if (size_sum_is_bigger_than(offset, extend, size)) {
1864 rb_raise(rb_eArgError,
"Type extends beyond end of buffer! (offset=%"PRIdSIZE
" > size=%"PRIdSIZE
")", offset, size);
1889#define ruby_swap8(value) value
1897ruby_swapf32(
float value)
1899 union swapf32 swap = {.value = value};
1900 swap.integral = ruby_swap32(swap.integral);
1910ruby_swapf64(
double value)
1912 union swapf64 swap = {.value = value};
1913 swap.integral = ruby_swap64(swap.integral);
1925#ifdef HAVE_UINT128_T
1926#if __has_builtin(__builtin_bswap128)
1927 result.value = __builtin_bswap128(x.value);
1930 uint64_t low = (uint64_t)x.value;
1931 uint64_t high = (uint64_t)(x.value >> 64);
1932 low = ruby_swap64(low);
1933 high = ruby_swap64(high);
1934 result.value = ((uint128_t)low << 64) | high;
1941 result.parts.low = ruby_swap64(x.parts.high);
1942 result.parts.high = ruby_swap64(x.parts.low);
1953 conversion.uint128 = ruby_swap128_uint(conversion.uint128);
1954 return conversion.int128;
1957#define IO_BUFFER_DECLARE_TYPE(name, type, endian, wrap, unwrap, swap) \
1958static ID RB_IO_BUFFER_DATA_TYPE_##name; \
1961io_buffer_read_##name(const void* base, size_t size, size_t *offset) \
1963 io_buffer_validate_type(size, *offset, sizeof(type)); \
1965 memcpy(&value, (char*)base + *offset, sizeof(type)); \
1966 if (endian != RB_IO_BUFFER_HOST_ENDIAN) value = swap(value); \
1967 *offset += sizeof(type); \
1968 return wrap(value); \
1972io_buffer_write_##name(const void* base, size_t size, size_t *offset, VALUE _value) \
1974 io_buffer_validate_type(size, *offset, sizeof(type)); \
1975 type value = unwrap(_value); \
1976 if (endian != RB_IO_BUFFER_HOST_ENDIAN) value = swap(value); \
1977 memcpy((char*)base + *offset, &value, sizeof(type)); \
1978 *offset += sizeof(type); \
1982 RB_IO_BUFFER_DATA_TYPE_##name##_SIZE = sizeof(type) \
1990IO_BUFFER_DECLARE_TYPE(s16, int16_t, RB_IO_BUFFER_LITTLE_ENDIAN,
RB_INT2NUM,
RB_NUM2INT, ruby_swap16)
1991IO_BUFFER_DECLARE_TYPE(S16, int16_t, RB_IO_BUFFER_BIG_ENDIAN,
RB_INT2NUM,
RB_NUM2INT, ruby_swap16)
1995IO_BUFFER_DECLARE_TYPE(s32, int32_t, RB_IO_BUFFER_LITTLE_ENDIAN,
RB_INT2NUM,
RB_NUM2INT, ruby_swap32)
1996IO_BUFFER_DECLARE_TYPE(S32, int32_t, RB_IO_BUFFER_BIG_ENDIAN,
RB_INT2NUM,
RB_NUM2INT, ruby_swap32)
1998IO_BUFFER_DECLARE_TYPE(u64, uint64_t, RB_IO_BUFFER_LITTLE_ENDIAN,
RB_ULL2NUM,
RB_NUM2ULL, ruby_swap64)
1999IO_BUFFER_DECLARE_TYPE(U64, uint64_t, RB_IO_BUFFER_BIG_ENDIAN,
RB_ULL2NUM,
RB_NUM2ULL, ruby_swap64)
2000IO_BUFFER_DECLARE_TYPE(s64, int64_t, RB_IO_BUFFER_LITTLE_ENDIAN,
RB_LL2NUM,
RB_NUM2LL, ruby_swap64)
2001IO_BUFFER_DECLARE_TYPE(S64, int64_t, RB_IO_BUFFER_BIG_ENDIAN,
RB_LL2NUM,
RB_NUM2LL, ruby_swap64)
2003IO_BUFFER_DECLARE_TYPE(u128,
rb_uint128_t, RB_IO_BUFFER_LITTLE_ENDIAN, rb_uint128_to_numeric, rb_numeric_to_uint128, ruby_swap128_uint)
2004IO_BUFFER_DECLARE_TYPE(U128,
rb_uint128_t, RB_IO_BUFFER_BIG_ENDIAN, rb_uint128_to_numeric, rb_numeric_to_uint128, ruby_swap128_uint)
2005IO_BUFFER_DECLARE_TYPE(s128,
rb_int128_t, RB_IO_BUFFER_LITTLE_ENDIAN, rb_int128_to_numeric, rb_numeric_to_int128, ruby_swap128_int)
2006IO_BUFFER_DECLARE_TYPE(S128,
rb_int128_t, RB_IO_BUFFER_BIG_ENDIAN, rb_int128_to_numeric, rb_numeric_to_int128, ruby_swap128_int)
2008IO_BUFFER_DECLARE_TYPE(f32,
float, RB_IO_BUFFER_LITTLE_ENDIAN,
DBL2NUM,
NUM2DBL, ruby_swapf32)
2009IO_BUFFER_DECLARE_TYPE(F32,
float, RB_IO_BUFFER_BIG_ENDIAN,
DBL2NUM,
NUM2DBL, ruby_swapf32)
2010IO_BUFFER_DECLARE_TYPE(f64,
double, RB_IO_BUFFER_LITTLE_ENDIAN,
DBL2NUM,
NUM2DBL, ruby_swapf64)
2011IO_BUFFER_DECLARE_TYPE(F64,
double, RB_IO_BUFFER_BIG_ENDIAN,
DBL2NUM,
NUM2DBL, ruby_swapf64)
2012#undef IO_BUFFER_DECLARE_TYPE
2015io_buffer_buffer_type_size(
ID buffer_type)
2017#define IO_BUFFER_DATA_TYPE_SIZE(name) if (buffer_type == RB_IO_BUFFER_DATA_TYPE_##name) return RB_IO_BUFFER_DATA_TYPE_##name##_SIZE;
2018 IO_BUFFER_DATA_TYPE_SIZE(U8)
2019 IO_BUFFER_DATA_TYPE_SIZE(S8)
2020 IO_BUFFER_DATA_TYPE_SIZE(u16)
2021 IO_BUFFER_DATA_TYPE_SIZE(U16)
2022 IO_BUFFER_DATA_TYPE_SIZE(s16)
2023 IO_BUFFER_DATA_TYPE_SIZE(S16)
2024 IO_BUFFER_DATA_TYPE_SIZE(u32)
2025 IO_BUFFER_DATA_TYPE_SIZE(U32)
2026 IO_BUFFER_DATA_TYPE_SIZE(s32)
2027 IO_BUFFER_DATA_TYPE_SIZE(S32)
2028 IO_BUFFER_DATA_TYPE_SIZE(u64)
2029 IO_BUFFER_DATA_TYPE_SIZE(U64)
2030 IO_BUFFER_DATA_TYPE_SIZE(s64)
2031 IO_BUFFER_DATA_TYPE_SIZE(S64)
2032 IO_BUFFER_DATA_TYPE_SIZE(u128)
2033 IO_BUFFER_DATA_TYPE_SIZE(U128)
2034 IO_BUFFER_DATA_TYPE_SIZE(s128)
2035 IO_BUFFER_DATA_TYPE_SIZE(S128)
2036 IO_BUFFER_DATA_TYPE_SIZE(f32)
2037 IO_BUFFER_DATA_TYPE_SIZE(F32)
2038 IO_BUFFER_DATA_TYPE_SIZE(f64)
2039 IO_BUFFER_DATA_TYPE_SIZE(F64)
2040#undef IO_BUFFER_DATA_TYPE_SIZE
2042 rb_raise(rb_eArgError,
"Invalid type name!");
2056io_buffer_size_of(
VALUE klass,
VALUE buffer_type)
2060 for (
long i = 0; i <
RARRAY_LEN(buffer_type); i++) {
2071rb_io_buffer_get_value(
const void* base,
size_t size,
ID buffer_type,
size_t *offset)
2073#define IO_BUFFER_GET_VALUE(name) if (buffer_type == RB_IO_BUFFER_DATA_TYPE_##name) return io_buffer_read_##name(base, size, offset);
2074 IO_BUFFER_GET_VALUE(U8)
2075 IO_BUFFER_GET_VALUE(S8)
2077 IO_BUFFER_GET_VALUE(u16)
2078 IO_BUFFER_GET_VALUE(U16)
2079 IO_BUFFER_GET_VALUE(s16)
2080 IO_BUFFER_GET_VALUE(S16)
2082 IO_BUFFER_GET_VALUE(u32)
2083 IO_BUFFER_GET_VALUE(U32)
2084 IO_BUFFER_GET_VALUE(s32)
2085 IO_BUFFER_GET_VALUE(S32)
2087 IO_BUFFER_GET_VALUE(u64)
2088 IO_BUFFER_GET_VALUE(U64)
2089 IO_BUFFER_GET_VALUE(s64)
2090 IO_BUFFER_GET_VALUE(S64)
2092 IO_BUFFER_GET_VALUE(u128)
2093 IO_BUFFER_GET_VALUE(U128)
2094 IO_BUFFER_GET_VALUE(s128)
2095 IO_BUFFER_GET_VALUE(S128)
2097 IO_BUFFER_GET_VALUE(f32)
2098 IO_BUFFER_GET_VALUE(F32)
2099 IO_BUFFER_GET_VALUE(f64)
2100 IO_BUFFER_GET_VALUE(F64)
2101#undef IO_BUFFER_GET_VALUE
2103 rb_raise(rb_eArgError,
"Invalid type name!");
2149 size_t offset = io_buffer_extract_offset(_offset);
2151 rb_io_buffer_get_bytes_for_reading(self, &base, &size);
2153 return rb_io_buffer_get_value(base, size,
RB_SYM2ID(
type), &offset);
2169 size_t offset = io_buffer_extract_offset(_offset);
2173 rb_io_buffer_get_bytes_for_reading(self, &base, &size);
2176 rb_raise(rb_eArgError,
"Argument buffer_types should be an array!");
2181 for (
long i = 0; i <
RARRAY_LEN(buffer_types); i++) {
2193io_buffer_extract_count(
VALUE argument)
2195 if (rb_int_negative_p(argument)) {
2196 rb_raise(rb_eArgError,
"Count can't be negative!");
2203io_buffer_extract_offset_count(
ID buffer_type,
size_t size,
int argc,
VALUE *argv,
size_t *offset,
size_t *count)
2206 *offset = io_buffer_extract_offset(argv[0]);
2213 *count = io_buffer_extract_count(argv[1]);
2216 if (*offset > size) {
2217 rb_raise(rb_eArgError,
"The given offset is bigger than the buffer size!");
2220 *count = (size - *offset) / io_buffer_buffer_type_size(buffer_type);
2241io_buffer_each(
int argc,
VALUE *argv,
VALUE self)
2248 rb_io_buffer_get_bytes_for_reading(self, &base, &size);
2255 buffer_type = RB_IO_BUFFER_DATA_TYPE_U8;
2258 size_t offset, count;
2259 io_buffer_extract_offset_count(buffer_type, size, argc-1, argv+1, &offset, &count);
2261 for (
size_t i = 0; i < count; i++) {
2262 size_t current_offset = offset;
2263 VALUE value = rb_io_buffer_get_value(base, size, buffer_type, &offset);
2281io_buffer_values(
int argc,
VALUE *argv,
VALUE self)
2286 rb_io_buffer_get_bytes_for_reading(self, &base, &size);
2293 buffer_type = RB_IO_BUFFER_DATA_TYPE_U8;
2296 size_t offset, count;
2297 io_buffer_extract_offset_count(buffer_type, size, argc-1, argv+1, &offset, &count);
2301 for (
size_t i = 0; i < count; i++) {
2302 VALUE value = rb_io_buffer_get_value(base, size, buffer_type, &offset);
2325io_buffer_each_byte(
int argc,
VALUE *argv,
VALUE self)
2332 rb_io_buffer_get_bytes_for_reading(self, &base, &size);
2334 size_t offset, count;
2335 io_buffer_extract_offset_count(RB_IO_BUFFER_DATA_TYPE_U8, size, argc, argv, &offset, &count);
2337 for (
size_t i = 0; i < count; i++) {
2338 unsigned char *value = (
unsigned char *)base + i + offset;
2346rb_io_buffer_set_value(
const void* base,
size_t size,
ID buffer_type,
size_t *offset,
VALUE value)
2348#define IO_BUFFER_SET_VALUE(name) if (buffer_type == RB_IO_BUFFER_DATA_TYPE_##name) {io_buffer_write_##name(base, size, offset, value); return;}
2349 IO_BUFFER_SET_VALUE(U8);
2350 IO_BUFFER_SET_VALUE(S8);
2352 IO_BUFFER_SET_VALUE(u16);
2353 IO_BUFFER_SET_VALUE(U16);
2354 IO_BUFFER_SET_VALUE(s16);
2355 IO_BUFFER_SET_VALUE(S16);
2357 IO_BUFFER_SET_VALUE(u32);
2358 IO_BUFFER_SET_VALUE(U32);
2359 IO_BUFFER_SET_VALUE(s32);
2360 IO_BUFFER_SET_VALUE(S32);
2362 IO_BUFFER_SET_VALUE(u64);
2363 IO_BUFFER_SET_VALUE(U64);
2364 IO_BUFFER_SET_VALUE(s64);
2365 IO_BUFFER_SET_VALUE(S64);
2367 IO_BUFFER_SET_VALUE(u128);
2368 IO_BUFFER_SET_VALUE(U128);
2369 IO_BUFFER_SET_VALUE(s128);
2370 IO_BUFFER_SET_VALUE(S128);
2372 IO_BUFFER_SET_VALUE(f32);
2373 IO_BUFFER_SET_VALUE(F32);
2374 IO_BUFFER_SET_VALUE(f64);
2375 IO_BUFFER_SET_VALUE(F64);
2376#undef IO_BUFFER_SET_VALUE
2378 rb_raise(rb_eArgError,
"Invalid type name!");
2416 size_t offset = io_buffer_extract_offset(_offset);
2418 rb_io_buffer_get_bytes_for_writing(self, &base, &size);
2420 rb_io_buffer_set_value(base, size,
RB_SYM2ID(
type), &offset, value);
2443 rb_raise(rb_eArgError,
"Argument buffer_types should be an array!");
2447 rb_raise(rb_eArgError,
"Argument values should be an array!");
2451 rb_raise(rb_eArgError,
"Argument buffer_types and values should have the same length!");
2454 size_t offset = io_buffer_extract_offset(_offset);
2458 rb_io_buffer_get_bytes_for_writing(self, &base, &size);
2460 for (
long i = 0; i <
RARRAY_LEN(buffer_types); i++) {
2463 rb_io_buffer_set_value(base, size,
RB_SYM2ID(
type), &offset, value);
2469static size_t IO_BUFFER_BLOCKING_SIZE = 1024*1024;
2472 unsigned char * destination;
2473 const unsigned char * source;
2478io_buffer_memmove_blocking(
void *data)
2482 memmove(arguments->destination, arguments->source, arguments->length);
2488io_buffer_memmove_unblock(
void *data)
2494io_buffer_memmove(
struct rb_io_buffer *buffer,
size_t offset,
const void *source_base,
size_t source_offset,
size_t source_size,
size_t length)
2498 io_buffer_get_bytes_for_writing(buffer, &base, &size);
2500 io_buffer_validate_range(buffer, offset, length);
2502 if (size_sum_is_bigger_than(source_offset, length, source_size)) {
2503 rb_raise(rb_eArgError,
"The computed source range exceeds the size of the source buffer!");
2507 .destination = (
unsigned char*)base+offset,
2508 .source = (
unsigned char*)source_base+source_offset,
2512 if (arguments.length >= IO_BUFFER_BLOCKING_SIZE) {
2514 }
else if (arguments.length != 0) {
2515 memmove(arguments.destination, arguments.source, arguments.length);
2521io_buffer_copy_from(
struct rb_io_buffer *buffer,
const void *source_base,
size_t source_size,
int argc,
VALUE *argv)
2525 size_t source_offset;
2529 offset = io_buffer_extract_offset(argv[0]);
2534 source_offset = io_buffer_extract_offset(argv[2]);
2536 if (source_offset > source_size) {
2537 rb_raise(rb_eArgError,
"The given source offset is bigger than the source itself!");
2545 if (argc >= 2 && !
RB_NIL_P(argv[1])) {
2546 length = io_buffer_extract_length(argv[1]);
2550 length = source_size - source_offset;
2553 io_buffer_memmove(buffer, offset, source_base, source_offset, source_size, length);
2576rb_io_buffer_initialize_copy(
VALUE self,
VALUE source)
2581 const void *source_base;
2584 rb_io_buffer_get_bytes_for_reading(source, &source_base, &source_size);
2586 io_buffer_initialize(self, buffer, NULL, source_size, io_flags_for_size(source_size),
Qnil);
2588 VALUE result = io_buffer_copy_from(buffer, source_base, source_size, 0, NULL);
2662io_buffer_copy(
int argc,
VALUE *argv,
VALUE self)
2669 VALUE source = argv[0];
2670 const void *source_base;
2673 rb_io_buffer_get_bytes_for_reading(source, &source_base, &source_size);
2675 VALUE result = io_buffer_copy_from(buffer, source_base, source_size, argc-1, argv+1);
2695io_buffer_get_string(
int argc,
VALUE *argv,
VALUE self)
2699 size_t offset, length;
2700 struct rb_io_buffer *buffer = io_buffer_extract_offset_length(self, argc, argv, &offset, &length);
2704 io_buffer_get_bytes_for_reading(buffer, &base, &size);
2708 encoding = rb_find_encoding(argv[2]);
2711 encoding = rb_ascii8bit_encoding();
2714 io_buffer_validate_range(buffer, offset, length);
2716 return rb_enc_str_new((
const char*)base + offset, length, encoding);
2743io_buffer_set_string(
int argc,
VALUE *argv,
VALUE self)
2752 const void *source_base = RSTRING_PTR(
string);
2753 size_t source_size = RSTRING_LEN(
string);
2755 VALUE result = io_buffer_copy_from(buffer, source_base, source_size, argc-1, argv+1);
2761rb_io_buffer_clear(
VALUE self, uint8_t value,
size_t offset,
size_t length)
2768 io_buffer_get_bytes_for_writing(buffer, &base, &size);
2770 io_buffer_validate_range(buffer, offset, length);
2772 memset((
char*)base + offset, value, length);
2807io_buffer_clear(
int argc,
VALUE *argv,
VALUE self)
2816 size_t offset, length;
2817 io_buffer_extract_offset_length(self, argc-1, argv+1, &offset, &length);
2819 rb_io_buffer_clear(self, value, offset, length);
2825io_buffer_default_size(
size_t page_size)
2828 const size_t platform_agnostic_default_size = 64*1024;
2831 const char *default_size = getenv(
"RUBY_IO_BUFFER_DEFAULT_SIZE");
2834 int value = atoi(default_size);
2842 if (platform_agnostic_default_size < page_size) {
2846 return platform_agnostic_default_size;
2852 rb_blocking_function_t *function;
2857io_buffer_blocking_region_begin(
VALUE _argument)
2861 return rb_io_blocking_region(argument->io, argument->function, argument->data);
2865io_buffer_blocking_region_ensure(
VALUE _argument)
2869 io_buffer_unlock(argument->buffer);
2875io_buffer_blocking_region(
VALUE io,
struct rb_io_buffer *buffer, rb_blocking_function_t *function,
void *data)
2877 struct rb_io *ioptr;
2883 .function = function,
2888 if (buffer->flags & RB_IO_BUFFER_LOCKED) {
2889 return io_buffer_blocking_region_begin((
VALUE)&argument);
2893 io_buffer_lock(buffer);
2895 return rb_ensure(io_buffer_blocking_region_begin, (
VALUE)&argument, io_buffer_blocking_region_ensure, (
VALUE)&argument);
2911io_buffer_read_internal(
void *_argument)
2917 ssize_t result = read(argument->descriptor, argument->base, argument->size);
2922 else if (result == 0) {
2928 if (total >= argument->length) {
2932 argument->base = argument->base + result;
2933 argument->size = argument->size - result;
2939rb_io_buffer_read(
VALUE self,
VALUE io,
size_t length,
size_t offset)
2944 if (scheduler !=
Qnil) {
2947 if (!UNDEF_P(result)) {
2955 io_buffer_validate_range(buffer, offset, length);
2961 io_buffer_get_bytes_for_writing(buffer, &base, &size);
2963 base = (
unsigned char*)base + offset;
2964 size = size - offset;
2967 .descriptor = descriptor,
2973 return io_buffer_blocking_region(io, buffer, io_buffer_read_internal, &argument);
3003io_buffer_read(
int argc,
VALUE *argv,
VALUE self)
3009 size_t length, offset;
3010 io_buffer_extract_length_offset(self, argc-1, argv+1, &length, &offset);
3012 return rb_io_buffer_read(self, io, length, offset);
3029io_buffer_pread_internal(
void *_argument)
3035 ssize_t result = pread(argument->descriptor, argument->base, argument->size, argument->offset);
3040 else if (result == 0) {
3046 if (total >= argument->length) {
3050 argument->base = argument->base + result;
3051 argument->size = argument->size - result;
3052 argument->offset = argument->offset + result;
3058rb_io_buffer_pread(
VALUE self,
VALUE io, rb_off_t from,
size_t length,
size_t offset)
3063 if (scheduler !=
Qnil) {
3066 if (!UNDEF_P(result)) {
3074 io_buffer_validate_range(buffer, offset, length);
3080 io_buffer_get_bytes_for_writing(buffer, &base, &size);
3082 base = (
unsigned char*)base + offset;
3083 size = size - offset;
3086 .descriptor = descriptor,
3093 return io_buffer_blocking_region(io, buffer, io_buffer_pread_internal, &argument);
3127io_buffer_pread(
int argc,
VALUE *argv,
VALUE self)
3134 size_t length, offset;
3135 io_buffer_extract_length_offset(self, argc-2, argv+2, &length, &offset);
3137 return rb_io_buffer_pread(self, io, from, length, offset);
3152io_buffer_write_internal(
void *_argument)
3158 ssize_t result = write(argument->descriptor, argument->base, argument->size);
3163 else if (result == 0) {
3169 if (total >= argument->length) {
3173 argument->base = argument->base + result;
3174 argument->size = argument->size - result;
3180rb_io_buffer_write(
VALUE self,
VALUE io,
size_t length,
size_t offset)
3185 if (scheduler !=
Qnil) {
3188 if (!UNDEF_P(result)) {
3196 io_buffer_validate_range(buffer, offset, length);
3202 io_buffer_get_bytes_for_reading(buffer, &base, &size);
3204 base = (
unsigned char*)base + offset;
3205 size = size - offset;
3208 .descriptor = descriptor,
3214 return io_buffer_blocking_region(io, buffer, io_buffer_write_internal, &argument);
3237io_buffer_write(
int argc,
VALUE *argv,
VALUE self)
3243 size_t length, offset;
3244 io_buffer_extract_length_offset(self, argc-1, argv+1, &length, &offset);
3246 return rb_io_buffer_write(self, io, length, offset);
3263io_buffer_pwrite_internal(
void *_argument)
3269 ssize_t result = pwrite(argument->descriptor, argument->base, argument->size, argument->offset);
3274 else if (result == 0) {
3280 if (total >= argument->length) {
3284 argument->base = argument->base + result;
3285 argument->size = argument->size - result;
3286 argument->offset = argument->offset + result;
3292rb_io_buffer_pwrite(
VALUE self,
VALUE io, rb_off_t from,
size_t length,
size_t offset)
3297 if (scheduler !=
Qnil) {
3300 if (!UNDEF_P(result)) {
3308 io_buffer_validate_range(buffer, offset, length);
3314 io_buffer_get_bytes_for_reading(buffer, &base, &size);
3316 base = (
unsigned char*)base + offset;
3317 size = size - offset;
3320 .descriptor = descriptor,
3335 return io_buffer_blocking_region(io, buffer, io_buffer_pwrite_internal, &argument);
3363io_buffer_pwrite(
int argc,
VALUE *argv,
VALUE self)
3370 size_t length, offset;
3371 io_buffer_extract_length_offset(self, argc-2, argv+2, &length, &offset);
3373 return rb_io_buffer_pwrite(self, io, from, length, offset);
3379 if (buffer->size == 0)
3380 rb_raise(rb_eIOBufferMaskError,
"Zero-length mask given!");
3384memory_and(
unsigned char * restrict output,
unsigned char * restrict base,
size_t size,
unsigned char * restrict mask,
size_t mask_size)
3386 for (
size_t offset = 0; offset < size; offset += 1) {
3387 output[offset] = base[offset] & mask[offset % mask_size];
3412 io_buffer_check_mask(mask_buffer);
3414 VALUE output = rb_io_buffer_new(NULL, buffer->size, io_flags_for_size(buffer->size));
3418 memory_and(output_buffer->base, buffer->base, buffer->size, mask_buffer->base, mask_buffer->size);
3424memory_or(
unsigned char * restrict output,
unsigned char * restrict base,
size_t size,
unsigned char * restrict mask,
size_t mask_size)
3426 for (
size_t offset = 0; offset < size; offset += 1) {
3427 output[offset] = base[offset] | mask[offset % mask_size];
3452 io_buffer_check_mask(mask_buffer);
3454 VALUE output = rb_io_buffer_new(NULL, buffer->size, io_flags_for_size(buffer->size));
3458 memory_or(output_buffer->base, buffer->base, buffer->size, mask_buffer->base, mask_buffer->size);
3464memory_xor(
unsigned char * restrict output,
unsigned char * restrict base,
size_t size,
unsigned char * restrict mask,
size_t mask_size)
3466 for (
size_t offset = 0; offset < size; offset += 1) {
3467 output[offset] = base[offset] ^ mask[offset % mask_size];
3492 io_buffer_check_mask(mask_buffer);
3494 VALUE output = rb_io_buffer_new(NULL, buffer->size, io_flags_for_size(buffer->size));
3498 memory_xor(output_buffer->base, buffer->base, buffer->size, mask_buffer->base, mask_buffer->size);
3504memory_not(
unsigned char * restrict output,
unsigned char * restrict base,
size_t size)
3506 for (
size_t offset = 0; offset < size; offset += 1) {
3507 output[offset] = ~base[offset];
3524io_buffer_not(
VALUE self)
3529 VALUE output = rb_io_buffer_new(NULL, buffer->size, io_flags_for_size(buffer->size));
3533 memory_not(output_buffer->base, buffer->base, buffer->size);
3541 if (a->base > b->base) {
3542 return io_buffer_overlaps(b, a);
3545 return (b->base >= a->base) && (b->base < (
void*)((
unsigned char *)a->base + a->size));
3551 if (io_buffer_overlaps(a, b))
3552 rb_raise(rb_eIOBufferMaskError,
"Mask overlaps source buffer!");
3556memory_and_inplace(
unsigned char * restrict base,
size_t size,
unsigned char * restrict mask,
size_t mask_size)
3558 for (
size_t offset = 0; offset < size; offset += 1) {
3559 base[offset] &= mask[offset % mask_size];
3589 io_buffer_check_mask(mask_buffer);
3590 io_buffer_check_overlaps(buffer, mask_buffer);
3594 io_buffer_get_bytes_for_writing(buffer, &base, &size);
3596 memory_and_inplace(base, size, mask_buffer->base, mask_buffer->size);
3602memory_or_inplace(
unsigned char * restrict base,
size_t size,
unsigned char * restrict mask,
size_t mask_size)
3604 for (
size_t offset = 0; offset < size; offset += 1) {
3605 base[offset] |= mask[offset % mask_size];
3635 io_buffer_check_mask(mask_buffer);
3636 io_buffer_check_overlaps(buffer, mask_buffer);
3640 io_buffer_get_bytes_for_writing(buffer, &base, &size);
3642 memory_or_inplace(base, size, mask_buffer->base, mask_buffer->size);
3648memory_xor_inplace(
unsigned char * restrict base,
size_t size,
unsigned char * restrict mask,
size_t mask_size)
3650 for (
size_t offset = 0; offset < size; offset += 1) {
3651 base[offset] ^= mask[offset % mask_size];
3681 io_buffer_check_mask(mask_buffer);
3682 io_buffer_check_overlaps(buffer, mask_buffer);
3686 io_buffer_get_bytes_for_writing(buffer, &base, &size);
3688 memory_xor_inplace(base, size, mask_buffer->base, mask_buffer->size);
3694memory_not_inplace(
unsigned char * restrict base,
size_t size)
3696 for (
size_t offset = 0; offset < size; offset += 1) {
3697 base[offset] = ~base[offset];
3719io_buffer_not_inplace(
VALUE self)
3726 io_buffer_get_bytes_for_writing(buffer, &base, &size);
3728 memory_not_inplace(base, size);
3846 GetSystemInfo(&info);
3847 RUBY_IO_BUFFER_PAGE_SIZE = info.dwPageSize;
3849 RUBY_IO_BUFFER_PAGE_SIZE = sysconf(_SC_PAGESIZE);
3852 RUBY_IO_BUFFER_DEFAULT_SIZE = io_buffer_default_size(RUBY_IO_BUFFER_PAGE_SIZE);
3855 rb_define_const(rb_cIOBuffer,
"PAGE_SIZE",
SIZET2NUM(RUBY_IO_BUFFER_PAGE_SIZE));
3860 rb_define_const(rb_cIOBuffer,
"DEFAULT_SIZE",
SIZET2NUM(RUBY_IO_BUFFER_DEFAULT_SIZE));
3865 rb_define_method(rb_cIOBuffer,
"initialize_copy", rb_io_buffer_initialize_copy, 1);
3875 rb_define_const(rb_cIOBuffer,
"EXTERNAL",
RB_INT2NUM(RB_IO_BUFFER_EXTERNAL));
3878 rb_define_const(rb_cIOBuffer,
"INTERNAL",
RB_INT2NUM(RB_IO_BUFFER_INTERNAL));
3881 rb_define_const(rb_cIOBuffer,
"MAPPED",
RB_INT2NUM(RB_IO_BUFFER_MAPPED));
3884 rb_define_const(rb_cIOBuffer,
"SHARED",
RB_INT2NUM(RB_IO_BUFFER_SHARED));
3887 rb_define_const(rb_cIOBuffer,
"LOCKED",
RB_INT2NUM(RB_IO_BUFFER_LOCKED));
3890 rb_define_const(rb_cIOBuffer,
"PRIVATE",
RB_INT2NUM(RB_IO_BUFFER_PRIVATE));
3893 rb_define_const(rb_cIOBuffer,
"READONLY",
RB_INT2NUM(RB_IO_BUFFER_READONLY));
3896 rb_define_const(rb_cIOBuffer,
"LITTLE_ENDIAN",
RB_INT2NUM(RB_IO_BUFFER_LITTLE_ENDIAN));
3899 rb_define_const(rb_cIOBuffer,
"BIG_ENDIAN",
RB_INT2NUM(RB_IO_BUFFER_BIG_ENDIAN));
3902 rb_define_const(rb_cIOBuffer,
"HOST_ENDIAN",
RB_INT2NUM(RB_IO_BUFFER_HOST_ENDIAN));
3905 rb_define_const(rb_cIOBuffer,
"NETWORK_ENDIAN",
RB_INT2NUM(RB_IO_BUFFER_NETWORK_ENDIAN));
3931#define IO_BUFFER_DEFINE_DATA_TYPE(name) RB_IO_BUFFER_DATA_TYPE_##name = rb_intern_const(#name)
3932 IO_BUFFER_DEFINE_DATA_TYPE(U8);
3933 IO_BUFFER_DEFINE_DATA_TYPE(S8);
3935 IO_BUFFER_DEFINE_DATA_TYPE(u16);
3936 IO_BUFFER_DEFINE_DATA_TYPE(U16);
3937 IO_BUFFER_DEFINE_DATA_TYPE(s16);
3938 IO_BUFFER_DEFINE_DATA_TYPE(S16);
3940 IO_BUFFER_DEFINE_DATA_TYPE(u32);
3941 IO_BUFFER_DEFINE_DATA_TYPE(U32);
3942 IO_BUFFER_DEFINE_DATA_TYPE(s32);
3943 IO_BUFFER_DEFINE_DATA_TYPE(S32);
3945 IO_BUFFER_DEFINE_DATA_TYPE(u64);
3946 IO_BUFFER_DEFINE_DATA_TYPE(U64);
3947 IO_BUFFER_DEFINE_DATA_TYPE(s64);
3948 IO_BUFFER_DEFINE_DATA_TYPE(S64);
3950 IO_BUFFER_DEFINE_DATA_TYPE(u128);
3951 IO_BUFFER_DEFINE_DATA_TYPE(U128);
3952 IO_BUFFER_DEFINE_DATA_TYPE(s128);
3953 IO_BUFFER_DEFINE_DATA_TYPE(S128);
3955 IO_BUFFER_DEFINE_DATA_TYPE(f32);
3956 IO_BUFFER_DEFINE_DATA_TYPE(F32);
3957 IO_BUFFER_DEFINE_DATA_TYPE(f64);
3958 IO_BUFFER_DEFINE_DATA_TYPE(F64);
3959#undef IO_BUFFER_DEFINE_DATA_TYPE
#define rb_define_method(klass, mid, func, arity)
Defines klass#mid.
#define rb_define_singleton_method(klass, mid, func, arity)
Defines klass.mid.
static bool RB_OBJ_FROZEN(VALUE obj)
Checks if an object is frozen.
void rb_include_module(VALUE klass, VALUE module)
Includes a module to a class.
VALUE rb_define_class_under(VALUE outer, const char *name, VALUE super)
Defines a class under the namespace of outer.
int rb_block_given_p(void)
Determines if the current method is given a block.
#define T_STRING
Old name of RUBY_T_STRING.
#define OBJ_FROZEN
Old name of RB_OBJ_FROZEN.
#define rb_str_cat2
Old name of rb_str_cat_cstr.
#define CLASS_OF
Old name of rb_class_of.
#define SIZET2NUM
Old name of RB_SIZE2NUM.
#define NUM2UINT
Old name of RB_NUM2UINT.
#define NUM2DBL
Old name of rb_num2dbl.
#define Qnil
Old name of RUBY_Qnil.
#define T_ARRAY
Old name of RUBY_T_ARRAY.
#define NIL_P
Old name of RB_NIL_P.
#define DBL2NUM
Old name of rb_float_new.
#define NUM2SIZET
Old name of RB_NUM2SIZE.
void rb_category_warn(rb_warning_category_t category, const char *fmt,...)
Identical to rb_category_warning(), except it reports unless $VERBOSE is nil.
VALUE rb_eRuntimeError
RuntimeError exception.
@ RB_WARN_CATEGORY_EXPERIMENTAL
Warning is for experimental features.
VALUE rb_cObject
Object class.
static VALUE rb_class_of(VALUE obj)
Object to class mapping function.
VALUE rb_mComparable
Comparable module.
#define RB_OBJ_WRITE(old, slot, young)
Declaration of a "back" pointer.
VALUE rb_ary_new_capa(long capa)
Identical to rb_ary_new(), except it additionally specifies how many rooms of objects it should alloc...
VALUE rb_ary_push(VALUE ary, VALUE elem)
Special case of rb_ary_cat() that it adds only one element.
VALUE rb_ary_entry(VALUE ary, long off)
Queries an element of an array.
#define RETURN_ENUMERATOR_KW(obj, argc, argv, kw_splat)
Identical to RETURN_SIZED_ENUMERATOR_KW(), except its size is unknown.
static int rb_check_arity(int argc, int min, int max)
Ensures that the passed integer is in the passed range.
VALUE rb_str_append(VALUE dst, VALUE src)
Identical to rb_str_buf_append(), except it converts the right hand side before concatenating.
#define rb_str_new(str, len)
Allocates an instance of rb_cString.
VALUE rb_str_locktmp(VALUE str)
Obtains a "temporary lock" of the string.
VALUE rb_str_unlocktmp(VALUE str)
Releases a lock formerly obtained by rb_str_locktmp().
VALUE rb_str_buf_new(long capa)
Allocates a "string buffer".
#define rb_str_new_cstr(str)
Identical to rb_str_new, except it assumes the passed pointer is a pointer to a C string.
VALUE rb_class_name(VALUE obj)
Queries the name of the given object's class.
void rb_define_alloc_func(VALUE klass, rb_alloc_func_t func)
Sets the allocator function of a class.
#define RB_SYM2ID
Just another name of rb_sym2id.
VALUE rb_io_get_io(VALUE io)
Identical to rb_io_check_io(), except it raises exceptions on conversion failures.
int rb_io_descriptor(VALUE io)
Returns an integer representing the numeric file descriptor for io.
#define RB_IO_POINTER(obj, fp)
Queries the underlying IO pointer.
VALUE rb_io_get_write_io(VALUE io)
Queries the tied IO for writing.
void * rb_nogvl(void *(*func)(void *), void *data1, rb_unblock_function_t *ubf, void *data2, int flags)
Identical to rb_thread_call_without_gvl(), except it additionally takes "flags" that change the behav...
#define RB_NOGVL_OFFLOAD_SAFE
Passing this flag to rb_nogvl() indicates that the passed function is safe to offload to a background...
#define RB_NUM2INT
Just another name of rb_num2int_inline.
#define RB_UINT2NUM
Just another name of rb_uint2num_inline.
#define RB_INT2NUM
Just another name of rb_int2num_inline.
static unsigned int RB_NUM2UINT(VALUE x)
Converts an instance of rb_cNumeric into C's unsigned int.
#define RB_LL2NUM
Just another name of rb_ll2num_inline.
#define RB_ULL2NUM
Just another name of rb_ull2num_inline.
#define RB_NUM2ULL
Just another name of rb_num2ull_inline.
#define RB_NUM2LL
Just another name of rb_num2ll_inline.
VALUE rb_yield_values(int n,...)
Identical to rb_yield(), except it takes variadic number of parameters and pass them to the block.
VALUE rb_yield(VALUE val)
Yields the block.
static VALUE RB_INT2FIX(long i)
Converts a C's long into an instance of rb_cInteger.
#define RB_NUM2LONG
Just another name of rb_num2long_inline.
#define RB_GC_GUARD(v)
Prevents premature destruction of local objects.
VALUE type(ANYARGS)
ANYARGS-ed function type.
VALUE rb_ensure(type *q, VALUE w, type *e, VALUE r)
An equivalent of ensure clause.
#define NUM2OFFT
Converts an instance of rb_cNumeric into C's off_t.
#define RARRAY_LEN
Just another name of rb_array_len.
#define RARRAY_AREF(a, i)
#define StringValue(v)
Ensures that the parameter object is a String.
#define RSTRING_GETMEM(str, ptrvar, lenvar)
Convenient macro to obtain the contents and length at once.
VALUE rb_str_to_str(VALUE obj)
Identical to rb_check_string_type(), except it raises exceptions in case of conversion failures.
#define RUBY_TYPED_FREE_IMMEDIATELY
Macros to see if each corresponding flag is defined.
#define TypedData_Get_Struct(obj, type, data_type, sval)
Obtains a C struct from inside of a wrapper Ruby object.
#define TypedData_Make_Struct(klass, type, data_type, sval)
Identical to TypedData_Wrap_Struct, except it allocates a new data region internally instead of takin...
#define errno
Ractor-aware version of errno.
#define RB_NO_KEYWORDS
Do not pass keywords.
VALUE rb_fiber_scheduler_current(void)
Identical to rb_fiber_scheduler_get(), except it also returns RUBY_Qnil in case of a blocking fiber.
VALUE rb_fiber_scheduler_io_pwrite(VALUE scheduler, VALUE io, rb_off_t from, VALUE buffer, size_t length, size_t offset)
Non-blocking write to the passed IO at the specified offset.
VALUE rb_fiber_scheduler_io_read(VALUE scheduler, VALUE io, VALUE buffer, size_t length, size_t offset)
Non-blocking read from the passed IO.
static VALUE rb_fiber_scheduler_io_result(ssize_t result, int error)
Wrap a ssize_t and int errno into a single VALUE.
VALUE rb_fiber_scheduler_io_pread(VALUE scheduler, VALUE io, rb_off_t from, VALUE buffer, size_t length, size_t offset)
Non-blocking read from the passed IO at the specified offset.
VALUE rb_fiber_scheduler_io_write(VALUE scheduler, VALUE io, VALUE buffer, size_t length, size_t offset)
Non-blocking write to the passed IO.
static bool RB_NIL_P(VALUE obj)
Checks if the given object is nil.
This is the struct that holds necessary info for a struct.
const char * wrap_struct_name
Name of structs of this kind.
Ruby's IO, metadata and buffers.
uintptr_t ID
Type that represents a Ruby identifier such as a variable name.
uintptr_t VALUE
Type that represents a Ruby object.
static bool RB_TYPE_P(VALUE obj, enum ruby_value_type t)
Queries if the given object is of given type.