9#include "internal/bits.h"
10#include "internal/error.h"
11#include "internal/hash.h"
12#include "internal/proc.h"
13#include "internal/sanitizers.h"
14#include "internal/set_table.h"
15#include "internal/symbol.h"
16#include "internal/variable.h"
17#include "ruby_assert.h"
30#include "internal/gc.h"
36 union {
double d; st_index_t i;} u;
41static const uint64_t prime1 = ((uint64_t)0x2e0bb864 << 32) | 0xe9ea7df5;
42static const uint32_t prime2 = 0x830fcab9;
45mult_and_mix(uint64_t m1, uint64_t m2)
47#if defined HAVE_UINT128_T
48 uint128_t r = (uint128_t) m1 * (uint128_t) m2;
49 return (uint64_t) (r >> 64) ^ (uint64_t) r;
51 uint64_t hm1 = m1 >> 32, hm2 = m2 >> 32;
52 uint64_t lm1 = m1, lm2 = m2;
53 uint64_t v64_128 = hm1 * hm2;
54 uint64_t v32_96 = hm1 * lm2 + lm1 * hm2;
55 uint64_t v1_32 = lm1 * lm2;
57 return (v64_128 + (v32_96 >> 32)) ^ ((v32_96 << 32) + v1_32);
62key64_hash(uint64_t key, uint32_t seed)
64 return mult_and_mix(key + seed, prime1);
68#define set_index_hash(index) key64_hash(rb_hash_start(index), prime2)
71set_ident_hash(st_data_t n)
80 n ^= dbl_to_index(rb_float_value(n));
84 return (st_index_t)set_index_hash((st_index_t)n);
100static ID id_each_entry;
104static ID id_set_iter_lev;
105static ID id_subclass_compatible;
106static ID id_class_methods;
108#define RSET_INITIALIZED FL_USER1
109#define RSET_LEV_MASK (FL_USER13 | FL_USER14 | FL_USER15 | \
110 FL_USER16 | FL_USER17 | FL_USER18 | FL_USER19)
111#define RSET_LEV_SHIFT (FL_USHIFT + 13)
112#define RSET_LEV_MAX 127
114#define SET_ASSERT(expr) RUBY_ASSERT_MESG_WHEN(SET_DEBUG, expr, #expr)
116#define RSET_SIZE(set) set_table_size(RSET_TABLE(set))
117#define RSET_EMPTY(set) (RSET_SIZE(set) == 0)
118#define RSET_SIZE_NUM(set) SIZET2NUM(RSET_SIZE(set))
119#define RSET_IS_MEMBER(sobj, item) set_table_lookup(RSET_TABLE(set), (st_data_t)(item))
120#define RSET_COMPARE_BY_IDENTITY(set) (RSET_TABLE(set)->type == &identhash)
127mark_key(st_data_t key, st_data_t data)
129 rb_gc_mark_movable((
VALUE)key);
138 if (sobj->table.
entries) set_table_foreach(&sobj->table, mark_key, 0);
144 free((&sobj->table)->entries);
151 set_free_embedded(sobj);
152 memset(&sobj->table, 0,
sizeof(sobj->table));
156set_size(
const void *ptr)
160 return (
unsigned long)set_memsize(&sobj->table) -
sizeof(sobj->table);
164set_foreach_replace(st_data_t key, st_data_t argp,
int error)
166 if (rb_gc_location((
VALUE)key) != (
VALUE)key) {
174set_replace_ref(st_data_t *key, st_data_t argp,
int existing)
176 rb_gc_mark_and_move((
VALUE *)key);
182set_update_references(
void *ptr)
185 set_foreach_with_replace(&sobj->table, set_foreach_replace, set_replace_ref, 0);
194 .dcompact = set_update_references,
196 .flags = RUBY_TYPED_EMBEDDABLE | RUBY_TYPED_FREE_IMMEDIATELY | RUBY_TYPED_WB_PROTECTED | RUBY_TYPED_FROZEN_SHAREABLE
208iter_lev_in_ivar(
VALUE set)
213 SET_ASSERT(lev >= 0);
214 return (
unsigned long)lev;
220iter_lev_in_ivar_set(
VALUE set,
unsigned long lev)
222 SET_ASSERT(lev >= RSET_LEV_MAX);
224 rb_ivar_set_internal(set, id_set_iter_lev,
LONG2FIX((
long)lev));
227static inline unsigned long
228iter_lev_in_flags(
VALUE set)
230 return (
unsigned long)((
RBASIC(set)->flags >> RSET_LEV_SHIFT) & RSET_LEV_MAX);
234iter_lev_in_flags_set(
VALUE set,
unsigned long lev)
236 SET_ASSERT(lev <= RSET_LEV_MAX);
237 RBASIC(set)->flags = ((
RBASIC(set)->flags & ~RSET_LEV_MASK) | ((
VALUE)lev << RSET_LEV_SHIFT));
241set_iterating_p(
VALUE set)
243 return iter_lev_in_flags(set) > 0;
247set_iter_lev_inc(
VALUE set)
249 unsigned long lev = iter_lev_in_flags(set);
250 if (lev == RSET_LEV_MAX) {
251 lev = iter_lev_in_ivar(set) + 1;
258 iter_lev_in_flags_set(set, lev);
259 if (lev < RSET_LEV_MAX)
return;
261 iter_lev_in_ivar_set(set, lev);
265set_iter_lev_dec(
VALUE set)
267 unsigned long lev = iter_lev_in_flags(set);
268 if (lev == RSET_LEV_MAX) {
269 lev = iter_lev_in_ivar(set);
270 if (lev > RSET_LEV_MAX) {
271 iter_lev_in_ivar_set(set, lev-1);
274 rb_attr_delete(set, id_set_iter_lev);
279 iter_lev_in_flags_set(set, lev - 1);
283set_foreach_ensure(
VALUE set)
285 set_iter_lev_dec(set);
293 set_foreach_func *func;
298set_iter_status_check(
int status)
300 if (status == ST_CONTINUE) {
308set_foreach_iter(st_data_t key, st_data_t argp,
int error)
312 if (error)
return ST_STOP;
315 int status = (*arg->func)((
VALUE)key, arg->arg);
317 if (RSET_TABLE(arg->set) != tbl) {
321 return set_iter_status_check(status);
325set_foreach_call(
VALUE arg)
329 ret = set_foreach_check(RSET_TABLE(set), set_foreach_iter,
330 (st_data_t)arg, (st_data_t)
Qundef);
338set_iter(
VALUE set, set_foreach_func *func,
VALUE farg)
348 set_foreach_call((
VALUE)&arg);
351 set_iter_lev_inc(set);
352 rb_ensure(set_foreach_call, (
VALUE)&arg, set_foreach_ensure, set);
356NORETURN(
static void no_new_item(
void));
360 rb_raise(
rb_eRuntimeError,
"can't add a new item into set during iteration");
364set_compact_after_delete(
VALUE set)
366 if (!set_iterating_p(set)) {
367 set_compact_table(RSET_TABLE(set));
375 key = rb_hash_key_str(key);
376 if (key_addr) *key_addr = key;
378 int ret = set_insert(tab, (st_data_t)key);
386 return set_table_insert_wb(RSET_TABLE(set), set, key, key_addr);
390set_alloc_with_size(
VALUE klass, st_index_t size)
396 set_init_table_with_size(&sobj->table, &objhash, size);
403set_s_alloc(
VALUE klass)
405 return set_alloc_with_size(klass, 0);
416set_s_create(
int argc,
VALUE *argv,
VALUE klass)
418 VALUE set = set_alloc_with_size(klass, argc);
422 for (i=0; i < argc; i++) {
423 set_table_insert_wb(table, set, argv[i], NULL);
446 rb_raise(rb_eArgError,
"value must be a set");
451enum_method_id(
VALUE other)
454 return id_each_entry;
460 rb_raise(rb_eArgError,
"value must be enumerable");
467 return RSET_SIZE_NUM(set);
474 set_insert_wb(set, element, &element);
482 set_insert_wb(set, element, &element);
505set_i_initialize(
int argc,
VALUE *argv,
VALUE set)
507 if (
RBASIC(set)->flags & RSET_INITIALIZED) {
510 RBASIC(set)->flags |= RSET_INITIALIZED;
515 if (argc > 0 && (other = argv[0]) !=
Qnil) {
522 if (block_given) key =
rb_yield(key);
523 set_table_insert_wb(into, set, key, NULL);
528 rb_block_given_p() ? set_initialize_with_block : set_initialize_without_block,
540 if (set == other)
return set;
542 if (set_iterating_p(set)) {
549 set_free_embedded(sobj);
550 set_copy(&sobj->table, RSET_TABLE(other));
551 rb_gc_writebarrier_remember(set);
557set_inspect_i(st_data_t key, st_data_t arg)
561 if (args[1] ==
Qtrue) {
579 str = rb_sprintf(
"%"PRIsVALUE
"[...]", klass_name);
583 str = rb_sprintf(
"%"PRIsVALUE
"[", klass_name);
585 set_iter(set, set_inspect_i, (st_data_t)args);
607set_i_inspect(
VALUE set)
613set_to_a_i(st_data_t key, st_data_t arg)
631 st_index_t size = RSET_SIZE(set);
634 if (size == 0)
return ary;
636 if (ST_DATA_COMPATIBLE_P(
VALUE)) {
638 size = set_keys(RSET_TABLE(set), ptr, size);
640 rb_gc_writebarrier_remember(ary);
641 rb_ary_set_len(ary, size);
644 set_iter(set, set_to_a_i, (st_data_t)ary);
670 rb_warn_deprecated(
"passing arguments to Set#to_set", NULL);
710 rb_check_frozen(set);
711 if (set_iterating_p(set)) {
712 if (!set_table_lookup(RSET_TABLE(set), (st_data_t)item)) {
717 set_insert_wb(set, item, NULL);
736 rb_check_frozen(set);
737 if (set_iterating_p(set)) {
738 if (!set_table_lookup(RSET_TABLE(set), (st_data_t)item)) {
744 return set_insert_wb(set, item, NULL) ?
Qnil : set;
758 rb_check_frozen(set);
759 if (set_table_delete(RSET_TABLE(set), (st_data_t *)&item)) {
760 set_compact_after_delete(set);
775 rb_check_frozen(set);
776 if (set_table_delete(RSET_TABLE(set), (st_data_t *)&item)) {
777 set_compact_after_delete(set);
784set_delete_if_i(st_data_t key, st_data_t dummy)
798set_i_delete_if(
VALUE set)
801 rb_check_frozen(set);
802 set_iter(set, set_delete_if_i, 0);
803 set_compact_after_delete(set);
816set_i_reject(
VALUE set)
819 rb_check_frozen(set);
822 size_t n = set_table_size(table);
823 set_iter(set, set_delete_if_i, 0);
825 if (n == set_table_size(table))
return Qnil;
827 set_compact_after_delete(set);
832set_classify_i(st_data_t key, st_data_t tmp)
835 VALUE hash = args[0];
837 VALUE set = rb_hash_lookup2(hash, hash_key,
Qundef);
839 set = set_s_alloc(args[1]);
840 rb_hash_aset(hash, hash_key, set);
866set_i_classify(
VALUE set)
870 args[0] = rb_hash_new();
872 set_iter(set, set_classify_i, (st_data_t)args);
878set_divide_union_find_root(
long *uf_parents,
long index,
long *tmp_array)
880 long root = uf_parents[index];
881 long update_size = 0;
882 while (root != index) {
883 tmp_array[update_size++] = index;
885 root = uf_parents[index];
887 for (
long j = 0; j < update_size; j++) {
888 long idx = tmp_array[j];
889 uf_parents[idx] = root;
895set_divide_union_find_merge(
long *uf_parents,
long i,
long j,
long *tmp_array)
897 long root_i = set_divide_union_find_root(uf_parents, i, tmp_array);
898 long root_j = set_divide_union_find_root(uf_parents, j, tmp_array);
899 if (root_i != root_j) uf_parents[root_j] = root_i;
903set_divide_arity2(
VALUE set)
906 long size, *uf_parents, *tmp_array;
908 VALUE items = set_i_to_a(set);
911 tmp_array =
ALLOCV_N(
long, tmp, size);
912 uf_parents =
ALLOCV_N(
long, uf, size);
913 for (
long i = 0; i < size; i++) {
916 for (
long i = 0; i < size - 1; i++) {
918 for (
long j = i + 1; j < size; j++) {
922 set_divide_union_find_merge(uf_parents, i, j, tmp_array);
927 VALUE hash = rb_hash_new();
928 for (
long i = 0; i < size; i++) {
930 long root = set_divide_union_find_root(uf_parents, i, tmp_array);
933 set = set_s_create(0, 0, set_class);
934 rb_hash_aset(hash,
LONG2FIX(root), set);
935 set_i_add(final_set, set);
944static void set_merge_enum_into(
VALUE set,
VALUE arg);
970set_i_divide(
VALUE set)
974 if (rb_block_arity() == 2) {
975 return set_divide_arity2(set);
978 VALUE values = rb_hash_values(set_i_classify(set));
980 set_merge_enum_into(set, values);
985set_clear_i(st_data_t key, st_data_t dummy)
1001set_i_clear(
VALUE set)
1003 rb_check_frozen(set);
1004 if (RSET_SIZE(set) == 0)
return set;
1005 if (set_iterating_p(set)) {
1006 set_iter(set, set_clear_i, 0);
1009 set_table_clear(RSET_TABLE(set));
1010 set_compact_after_delete(set);
1022set_intersection_i(st_data_t key, st_data_t tmp)
1025 if (set_table_lookup(data->other, key)) {
1026 set_table_insert_wb(data->into, data->set, key, NULL);
1035 set_intersection_i((st_data_t)i, (st_data_t)data);
1054 set_table *ntable = RSET_TABLE(new_set);
1058 if (set_table_size(stable) >= set_table_size(otable)) {
1069 set_iter(set, set_intersection_i, (st_data_t)&data);
1077 rb_block_call(other, enum_method_id(other), 0, 0, set_intersection_block, (
VALUE)&data);
1110 return RBOOL(RSET_IS_MEMBER(set, item));
1119set_merge_i(st_data_t key, st_data_t data)
1122 set_table_insert_wb(args->into, args->set, key, NULL);
1129 VALUE element = key;
1130 set_insert_wb(set, element, &element);
1140 .into = RSET_TABLE(set)
1142 set_iter(arg, set_merge_i, (st_data_t)&args);
1148 set_table_insert_wb(into, set,
RARRAY_AREF(arg, i), NULL);
1167 rb_raise(rb_eArgError,
"no keywords accepted");
1170 if (set_iterating_p(set)) {
1174 rb_check_frozen(set);
1178 for (i=0; i < argc; i++) {
1179 set_merge_enum_into(set, argv[i]);
1188 rb_check_frozen(set);
1194 size_t size = set_table_size(old);
1196 set_table *
new = set_init_table_with_size(NULL,
type, size);
1201 set_iter(set, set_merge_i, (st_data_t)&args);
1202 set_free_embedded(sobj);
1203 memcpy(&sobj->table,
new,
sizeof(*
new));
1207 sobj->table.type =
type;
1220set_i_compare_by_identity(
VALUE set)
1222 if (RSET_COMPARE_BY_IDENTITY(set))
return set;
1224 if (set_iterating_p(set)) {
1228 return set_reset_table_with_type(set, &identhash);
1239set_i_compare_by_identity_p(
VALUE set)
1241 return RBOOL(RSET_COMPARE_BY_IDENTITY(set));
1251set_i_size(
VALUE set)
1253 return RSET_SIZE_NUM(set);
1263set_i_empty(
VALUE set)
1265 return RBOOL(RSET_EMPTY(set));
1269set_xor_i(st_data_t key, st_data_t data)
1274 if (set_table_insert_wb(table, set, element, &element)) {
1275 set_table_delete(table, &element);
1300 set_merge_enum_into(new_set, other);
1302 set_iter(set, set_xor_i, (st_data_t)new_set);
1320 set_merge_enum_into(set, other);
1325set_remove_i(st_data_t key, st_data_t from)
1327 set_table_delete((
struct set_table *)from, (st_data_t *)&key);
1334 rb_check_frozen(set);
1335 set_table_delete(RSET_TABLE(set), (st_data_t *)&key);
1343 set_iter(arg, set_remove_i, (st_data_t)RSET_TABLE(set));
1360 rb_check_frozen(set);
1361 set_remove_enum_from(set, other);
1378 return set_i_subtract(
rb_obj_dup(set), other);
1382set_each_i(st_data_t key, st_data_t dummy)
1398set_i_each(
VALUE set)
1401 set_iter(set, set_each_i, 0);
1406set_collect_i(st_data_t key, st_data_t data)
1421set_i_collect(
VALUE set)
1424 rb_check_frozen(set);
1427 set_iter(set, set_collect_i, (st_data_t)new_set);
1428 set_i_initialize_copy(set, new_set);
1434set_keep_if_i(st_data_t key, st_data_t into)
1437 set_table_delete((
set_table *)into, &key);
1451set_i_keep_if(
VALUE set)
1454 rb_check_frozen(set);
1456 set_iter(set, set_keep_if_i, (st_data_t)RSET_TABLE(set));
1470set_i_select(
VALUE set)
1473 rb_check_frozen(set);
1476 size_t n = set_table_size(table);
1477 set_iter(set, set_keep_if_i, (st_data_t)table);
1479 return (n == set_table_size(table)) ?
Qnil : set;
1496 rb_check_frozen(set);
1499 set_i_initialize_copy(set, other);
1502 if (set_iterating_p(set)) {
1507 enum_method_id(other);
1509 set_table_clear(RSET_TABLE(set));
1510 set_merge_enum_into(set, other);
1524set_i_reset(
VALUE set)
1526 if (set_iterating_p(set)) {
1530 return set_reset_table_with_type(set, RSET_TABLE(set)->
type);
1536set_flatten_merge_i(st_data_t item, st_data_t arg)
1539 VALUE set = args[0];
1541 VALUE e_id = rb_obj_id(item);
1542 VALUE hash = args[2];
1543 switch(rb_hash_aref(hash, e_id)) {
1547 rb_raise(rb_eArgError,
"tried to flatten recursive Set");
1552 rb_hash_aset(hash, e_id,
Qtrue);
1553 set_flatten_merge(set, item, hash);
1554 rb_hash_aset(hash, e_id,
Qfalse);
1557 set_i_add(set, item);
1565 VALUE args[3] = {set, from, hash};
1566 set_iter(from, set_flatten_merge_i, (st_data_t)args);
1577set_i_flatten(
VALUE set)
1580 set_flatten_merge(new_set, set, rb_hash_new());
1585set_contains_set_i(st_data_t item, st_data_t arg)
1588 *(
bool *)arg =
true;
1602set_i_flatten_bang(
VALUE set)
1604 bool contains_set =
false;
1605 set_iter(set, set_contains_set_i, (st_data_t)&contains_set);
1606 if (!contains_set)
return Qnil;
1607 rb_check_frozen(set);
1608 return set_i_replace(set, set_i_flatten(set));
1617set_le_i(st_data_t key, st_data_t arg)
1620 if (set_table_lookup(data->table, key))
return ST_CONTINUE;
1629 .table = RSET_TABLE(other),
1632 set_iter(set, set_le_i, (st_data_t)&data);
1646 if (RSET_SIZE(set) >= RSET_SIZE(other))
return Qfalse;
1647 return set_le(set, other);
1660 if (RSET_SIZE(set) > RSET_SIZE(other))
return Qfalse;
1661 return set_le(set, other);
1674 if (RSET_SIZE(set) <= RSET_SIZE(other))
return Qfalse;
1675 return set_le(other, set);
1688 if (RSET_SIZE(set) < RSET_SIZE(other))
return Qfalse;
1689 return set_le(other, set);
1693set_intersect_i(st_data_t key, st_data_t arg)
1696 if (set_table_lookup((
set_table *)args[0], key)) {
1719 size_t set_size = RSET_SIZE(set);
1720 size_t other_size = RSET_SIZE(other);
1725 if (set_size < other_size) {
1727 args[0] = (
VALUE)RSET_TABLE(other);
1731 args[0] = (
VALUE)RSET_TABLE(set);
1733 set_iter(iter_arg, set_intersect_i, (st_data_t)args);
1740 rb_raise(rb_eArgError,
"value must be enumerable");
1759 return RBOOL(!
RTEST(set_i_intersect(set, other)));
1774 size_t set_size = RSET_SIZE(set);
1775 size_t other_size = RSET_SIZE(other);
1777 if (set_size < other_size) {
1778 if (set_le(set, other) ==
Qtrue) {
1782 else if (set_size > other_size) {
1783 if (set_le(other, set) ==
Qtrue) {
1787 else if (set_le(set, other) ==
Qtrue) {
1801set_eql_i(st_data_t item, st_data_t arg)
1805 if (!set_table_lookup(RSET_TABLE(data->set), item)) {
1813set_recursive_eql(
VALUE set,
VALUE dt,
int recur)
1815 if (recur)
return Qtrue;
1817 data->result =
Qtrue;
1818 set_iter(set, set_eql_i, dt);
1819 return data->result;
1832 if (set == other)
return Qtrue;
1836 size_t ssize = set_table_size(stable);
1837 size_t osize = set_table_size(otable);
1839 if (ssize != osize)
return Qfalse;
1840 if (ssize == 0 && osize == 0)
return Qtrue;
1841 if (stable->type != otable->type)
return Qfalse;
1849set_hash_i(st_data_t item, st_data_t(arg))
1851 st_index_t *hval = (st_index_t *)arg;
1852 st_index_t ival = rb_hash(item);
1853 *hval ^= rb_st_hash(&ival,
sizeof(st_index_t), 0);
1864set_i_hash(
VALUE set)
1866 st_index_t size = RSET_SIZE(set);
1867 st_index_t hval = rb_st_hash_start(size);
1870 set_iter(set, set_hash_i, (
VALUE)&hval);
1872 hval = rb_st_hash_end(hval);
1878set_to_hash_i(st_data_t key, st_data_t arg)
1885set_i_to_h(
VALUE set)
1887 st_index_t size = RSET_SIZE(set);
1889 if (RSET_COMPARE_BY_IDENTITY(set)) {
1890 hash = rb_ident_hash_new_with_size(size);
1893 hash = rb_hash_new_with_size(size);
1895 rb_hash_set_default(hash,
Qfalse);
1897 if (size == 0)
return hash;
1899 set_iter(set, set_to_hash_i, (st_data_t)hash);
1904compat_dumper(
VALUE set)
1912set_i_from_hash_i(st_data_t key, st_data_t val, st_data_t set)
1925 if (rb_hash_compare_by_id_p(hash)) set_i_compare_by_identity(set);
1926 rb_hash_stlike_foreach(hash, set_i_from_hash_i, (st_data_t)set);
1933 return set_i_from_hash(self,
rb_ivar_get(a, id_i_hash));
1941 set_iter(set, func, arg);
1947 return set_alloc_with_size(
rb_cSet, 0);
1953 return set_alloc_with_size(
rb_cSet, (st_index_t)
capa);
1959 return RSET_IS_MEMBER(set, element);
1965 return set_i_add_p(set, element) !=
Qnil;
1971 return set_i_clear(set);
1977 return set_i_delete_p(set, element) !=
Qnil;
1983 return RSET_SIZE(set);
2207 id_set_iter_lev = rb_make_internal_id();
#define rb_define_method(klass, mid, func, arity)
Defines klass#mid.
#define rb_define_singleton_method(klass, mid, func, arity)
Defines klass.mid.
#define rb_define_private_method(klass, mid, func, arity)
Defines klass#mid and makes it private.
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(const char *name, VALUE super)
Defines a top-level class.
void rb_extend_object(VALUE obj, VALUE module)
Extend the object with the module.
VALUE rb_singleton_class(VALUE obj)
Finds or creates the singleton class of the passed object.
VALUE rb_define_class_under(VALUE outer, const char *name, VALUE super)
Defines a class under the namespace of outer.
void rb_define_alias(VALUE klass, const char *name1, const char *name2)
Defines an alias of a method.
int rb_keyword_given_p(void)
Determines if the current method is given a keyword argument.
int rb_block_given_p(void)
Determines if the current method is given a block.
#define rb_str_buf_cat2
Old name of rb_usascii_str_new_cstr.
#define Qundef
Old name of RUBY_Qundef.
#define CLASS_OF
Old name of rb_class_of.
#define LONG2FIX
Old name of RB_INT2FIX.
#define T_HASH
Old name of RUBY_T_HASH.
#define FLONUM_P
Old name of RB_FLONUM_P.
#define Qtrue
Old name of RUBY_Qtrue.
#define ST2FIX
Old name of RB_ST2FIX.
#define INT2NUM
Old name of RB_INT2NUM.
#define Qnil
Old name of RUBY_Qnil.
#define Qfalse
Old name of RUBY_Qfalse.
#define FIX2LONG
Old name of RB_FIX2LONG.
#define T_ARRAY
Old name of RUBY_T_ARRAY.
#define ALLOCV_N
Old name of RB_ALLOCV_N.
#define POSFIXABLE
Old name of RB_POSFIXABLE.
#define FIXNUM_P
Old name of RB_FIXNUM_P.
#define ALLOCV_END
Old name of RB_ALLOCV_END.
VALUE rb_eRuntimeError
RuntimeError exception.
VALUE rb_class_new_instance(int argc, const VALUE *argv, VALUE klass)
Allocates, then initialises an instance of the given class.
VALUE rb_mEnumerable
Enumerable module.
VALUE rb_obj_class(VALUE obj)
Queries the class of an object.
VALUE rb_obj_dup(VALUE obj)
Duplicates the given object.
VALUE rb_inspect(VALUE obj)
Generates a human-readable textual representation of the given object.
VALUE rb_obj_is_instance_of(VALUE obj, VALUE klass)
Queries if the given object is a direct instance of the given class.
VALUE rb_obj_is_kind_of(VALUE obj, VALUE klass)
Queries if the given object is an instance (of possibly descendants) of the given class.
VALUE rb_cString
String class.
#define RB_OBJ_WRITTEN(old, oldv, young)
Identical to RB_OBJ_WRITE(), except it doesn't write any values, but only a WB declaration.
VALUE rb_str_export_to_enc(VALUE obj, rb_encoding *enc)
Identical to rb_str_export(), except it additionally takes an encoding.
VALUE rb_funcall_passing_block(VALUE recv, ID mid, int argc, const VALUE *argv)
Identical to rb_funcallv_public(), except you can pass the passed block.
VALUE rb_funcall(VALUE recv, ID mid, int n,...)
Calls a method.
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_freeze(VALUE obj)
Freeze an array, preventing further modifications.
VALUE rb_ary_join(VALUE ary, VALUE sep)
Recursively stringises the elements of the passed array, flattens that result, then joins the sequenc...
#define RETURN_SIZED_ENUMERATOR(obj, argc, argv, size_fn)
This roughly resembles return enum_for(__callee__) unless block_given?.
static int rb_check_arity(int argc, int min, int max)
Ensures that the passed integer is in the passed range.
void rb_provide(const char *feature)
Declares that the given feature is already provided by someone else.
#define rb_hash_uint(h, i)
Just another name of st_hash_uint.
VALUE rb_str_buf_append(VALUE dst, VALUE src)
Identical to rb_str_cat_cstr(), except it takes Ruby's string instead of C's.
VALUE rb_str_buf_cat_ascii(VALUE dst, const char *src)
Identical to rb_str_cat_cstr(), except it additionally assumes the source string be a NUL terminated ...
VALUE rb_exec_recursive(VALUE(*f)(VALUE g, VALUE h, int r), VALUE g, VALUE h)
"Recursion" API entry point.
VALUE rb_exec_recursive_paired(VALUE(*f)(VALUE g, VALUE h, int r), VALUE g, VALUE p, VALUE h)
Identical to rb_exec_recursive(), except it checks for the recursion on the ordered pair of { g,...
VALUE rb_const_get(VALUE space, ID name)
Identical to rb_const_defined(), except it returns the actual defined value.
VALUE rb_ivar_set(VALUE obj, ID name, VALUE val)
Identical to rb_iv_set(), except it accepts the name as an ID instead of a C string.
VALUE rb_ivar_get(VALUE obj, ID name)
Identical to rb_iv_get(), except it accepts the name as an ID instead of a C string.
VALUE rb_class_path(VALUE mod)
Identical to rb_mod_name(), except it returns #<Class: ...> style inspection for anonymous modules.
int rb_respond_to(VALUE obj, ID mid)
Queries if the object responds to the method.
void rb_define_alloc_func(VALUE klass, rb_alloc_func_t func)
Sets the allocator function of a class.
static ID rb_intern_const(const char *str)
This is a "tiny optimisation" over rb_intern().
int capa
Designed capacity of the buffer.
#define RB_BLOCK_CALL_FUNC_ARGLIST(yielded_arg, callback_arg)
Shim for block function parameters.
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.
void rb_marshal_define_compat(VALUE newclass, VALUE oldclass, VALUE(*dumper)(VALUE), VALUE(*loader)(VALUE, VALUE))
Marshal format compatibility layer.
VALUE rb_block_call(VALUE q, ID w, int e, const VALUE *r, type *t, VALUE y)
Call a method with a block.
VALUE type(ANYARGS)
ANYARGS-ed function type.
VALUE rb_ensure(type *q, VALUE w, type *e, VALUE r)
An equivalent of ensure clause.
#define RARRAY_LEN
Just another name of rb_array_len.
#define RARRAY_PTR_USE(ary, ptr_name, expr)
Declares a section of code where raw pointers are used.
#define RARRAY_AREF(a, i)
#define RBASIC(obj)
Convenient casting macro.
#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...
VALUE rb_require(const char *feature)
Identical to rb_require_string(), except it takes C's string instead of Ruby's.
size_t rb_set_size(VALUE set)
Returns the number of elements in the set.
VALUE rb_set_clear(VALUE set)
Removes all entries from set.
bool rb_set_delete(VALUE set, VALUE element)
Removes the element from from set.
bool rb_set_add(VALUE set, VALUE element)
Adds element to set.
void rb_set_foreach(VALUE set, int(*func)(VALUE element, VALUE arg), VALUE arg)
Iterates over a set.
bool rb_set_lookup(VALUE set, VALUE element)
Whether the set contains the given element.
VALUE rb_set_new(void)
Creates a new, empty set object.
VALUE rb_set_new_capa(size_t capa)
Identical to rb_set_new(), except it additionally specifies how many elements it is expected to conta...
#define RTEST
This is an old name of RB_TEST.
This is the struct that holds necessary info for a struct.
const char * wrap_struct_name
Name of structs of this kind.
set_table_entry * entries
Array of size 2^entry_power.
uintptr_t ID
Type that represents a Ruby identifier such as a variable name.
uintptr_t VALUE
Type that represents a Ruby object.
static void Check_Type(VALUE v, enum ruby_value_type t)
Identical to RB_TYPE_P(), except it raises exceptions on predication failure.
static bool RB_TYPE_P(VALUE obj, enum ruby_value_type t)
Queries if the given object is of given type.