3#include "eval_intern.h"
5#include "internal/box.h"
6#include "internal/class.h"
7#include "internal/eval.h"
8#include "internal/error.h"
9#include "internal/file.h"
10#include "internal/gc.h"
11#include "internal/hash.h"
12#include "internal/io.h"
13#include "internal/load.h"
14#include "internal/st.h"
15#include "internal/variable.h"
30#ifdef HAVE_SYS_SENDFILE_H
31# include <sys/sendfile.h>
38VALUE rb_cBoxEntry = 0;
39VALUE rb_mBoxLoader = 0;
48static bool tmp_dir_has_dirsep;
50#define BOX_TMP_PREFIX "_ruby_box_"
53# define MAXPATHLEN 1024
62bool ruby_box_enabled =
false;
63bool ruby_box_init_done =
false;
64bool ruby_box_crashed =
false;
68static void cleanup_all_local_extensions(
VALUE libmap);
74 box_gem_flags->gem = flags->gem;
75 box_gem_flags->error_highlight = flags->error_highlight;
76 box_gem_flags->did_you_mean = flags->did_you_mean;
77 box_gem_flags->syntax_suggest = flags->syntax_suggest;
83 ruby_box_init_done =
true;
121 return rb_vm_current_box(GET_EC());
132 return rb_vm_loading_box(GET_EC());
136rb_current_box_in_crash_report(
void)
138 if (ruby_box_crashed)
140 return rb_current_box();
143static long box_id_counter = 0;
150 id = ++box_id_counter;
156box_main_to_s(
VALUE obj)
164 const rb_box_t *master = rb_master_box();
173 box->load_path =
rb_ary_dup(master->load_path);
174 box->expanded_load_path =
rb_ary_dup(master->expanded_load_path);
176 box->load_path_check_cache = 0;
177 box->loaded_features =
rb_ary_dup(master->loaded_features);
179 box->loaded_features_index = st_init_numtable();
180 box->loaded_features_realpaths = rb_hash_dup(master->loaded_features_realpaths);
181 box->loaded_features_realpath_map = rb_hash_dup(master->loaded_features_realpath_map);
182 box->loading_table = st_init_strtable();
183 box->ruby_dln_libmap = rb_hash_new_with_size(0);
184 box->gvar_tbl = rb_hash_new_with_size(0);
185 box->classext_cow_classes = st_init_numtable();
188 box->is_optional =
true;
192rb_box_gc_update_references(
void *ptr)
198 box->box_object = rb_gc_location(box->box_object);
200 box->top_self = rb_gc_location(box->top_self);
201 box->load_path = rb_gc_location(box->load_path);
202 box->expanded_load_path = rb_gc_location(box->expanded_load_path);
203 box->load_path_snapshot = rb_gc_location(box->load_path_snapshot);
204 if (box->load_path_check_cache) {
205 box->load_path_check_cache = rb_gc_location(box->load_path_check_cache);
207 box->loaded_features = rb_gc_location(box->loaded_features);
208 box->loaded_features_snapshot = rb_gc_location(box->loaded_features_snapshot);
209 box->loaded_features_realpaths = rb_gc_location(box->loaded_features_realpaths);
210 box->loaded_features_realpath_map = rb_gc_location(box->loaded_features_realpath_map);
211 box->ruby_dln_libmap = rb_gc_location(box->ruby_dln_libmap);
212 box->gvar_tbl = rb_gc_location(box->gvar_tbl);
216rb_box_entry_mark(
void *ptr)
221 rb_gc_mark(box->box_object);
222 rb_gc_mark(box->top_self);
223 rb_gc_mark(box->load_path);
224 rb_gc_mark(box->expanded_load_path);
225 rb_gc_mark(box->load_path_snapshot);
226 rb_gc_mark(box->load_path_check_cache);
227 rb_gc_mark(box->loaded_features);
228 rb_gc_mark(box->loaded_features_snapshot);
229 rb_gc_mark(box->loaded_features_realpaths);
230 rb_gc_mark(box->loaded_features_realpath_map);
231 if (box->loading_table) {
232 rb_mark_tbl(box->loading_table);
234 rb_gc_mark(box->ruby_dln_libmap);
235 rb_gc_mark(box->gvar_tbl);
236 if (box->classext_cow_classes) {
237 rb_mark_set(box->classext_cow_classes);
242free_loading_table_entry(st_data_t key, st_data_t value, st_data_t arg)
249free_loaded_feature_index_i(st_data_t key, st_data_t value, st_data_t arg)
252 rb_darray_free_sized((
void *)value,
long);
258free_box_st_tables(
void *ptr)
261 if (box->loading_table) {
262 st_foreach(box->loading_table, free_loading_table_entry, 0);
263 st_free_table(box->loading_table);
264 box->loading_table = 0;
267 if (box->loaded_features_index) {
268 st_foreach(box->loaded_features_index, free_loaded_feature_index_i, 0);
269 st_free_table(box->loaded_features_index);
274free_classext_for_box(st_data_t key, st_data_t _value, st_data_t box_arg)
281 ext = rb_class_unlink_classext(obj, box);
282 rb_class_classext_free(obj, ext,
false);
285 ext = rb_class_unlink_classext(obj, box);
286 rb_iclass_classext_free(obj, ext,
false);
289 rb_bug(
"Invalid type of object in classext_cow_classes: %s", rb_type_str(
BUILTIN_TYPE(obj)));
295box_entry_free(
void *ptr)
299 if (box->classext_cow_classes) {
300 st_foreach(box->classext_cow_classes, free_classext_for_box, (st_data_t)box);
303 cleanup_all_local_extensions(box->ruby_dln_libmap);
305 free_box_st_tables(ptr);
310box_entry_memsize(
const void *ptr)
314 if (box->loaded_features_index) {
315 size += rb_st_memsize(box->loaded_features_index);
317 if (box->loading_table) {
318 size += rb_st_memsize(box->loading_table);
329 rb_box_gc_update_references,
340 rb_box_gc_update_references,
346rb_box_entry_alloc(
VALUE klass)
350 box_entry_initialize(entry);
355get_box_struct_internal(
VALUE entry)
363rb_get_box_t(
VALUE box)
373 VM_ASSERT(BOX_OBJ_P(box));
375 CONST_ID(id_box_entry,
"__box_entry__");
376 entry = rb_attr_get(box, id_box_entry);
377 return get_box_struct_internal(entry);
383 VM_ASSERT(box && box->box_object);
384 return box->box_object;
394box_initialize(
VALUE box_value)
400 CONST_ID(id_box_entry,
"__box_entry__");
402 if (!rb_box_available()) {
403 rb_raise(
rb_eRuntimeError,
"Ruby Box is disabled. Set RUBY_BOX=1 environment variable to use Ruby::Box.");
407 box = get_box_struct_internal(entry);
409 box->box_object = box_value;
410 box->box_id = box_generate_id();
416 RCLASS_SET_PRIME_CLASSEXT_WRITABLE(box_value,
true);
420 object_classext = RCLASS_EXT_WRITABLE_IN_BOX(
rb_cObject, box);
421 RCLASS_SET_CONST_TBL(box_value, RCLASSEXT_CONST_TBL(object_classext),
true);
425 if (ruby_box_init_done) {
426 if (box_gem_flags->gem) {
427 rb_vm_call_cfunc_in_box(
Qnil, rb_define_gem_modules, (
VALUE)box_gem_flags,
Qnil,
429 rb_load_gem_prelude((
VALUE)box);
434 rb_zjit_invalidate_root_box();
446rb_box_s_getenabled(
VALUE recv)
448 return RBOOL(rb_box_available());
459rb_box_s_current(
VALUE recv)
463 if (!rb_box_available())
466 box = rb_vm_current_box(GET_EC());
467 VM_ASSERT(box && box->box_object);
468 return box->box_object;
478rb_box_load_path(
VALUE box)
480 VM_ASSERT(BOX_OBJ_P(box));
481 return rb_get_box_t(box)->load_path;
485UINT rb_w32_system_tmpdir(WCHAR *path, UINT
len);
490system_default_tmpdir(
void)
494 WCHAR tmppath[_MAX_PATH];
495 UINT
len = rb_w32_system_tmpdir(tmppath, numberof(tmppath));
497 int blen = WideCharToMultiByte(CP_UTF8, 0, tmppath,
len, NULL, 0, NULL, NULL);
498 char *tmpdir =
xmalloc(blen + 1);
499 WideCharToMultiByte(CP_UTF8, 0, tmppath,
len, tmpdir, blen, NULL, NULL);
503#elif defined _CS_DARWIN_USER_TEMP_DIR
504 char path[MAXPATHLEN];
505 size_t len = confstr(_CS_DARWIN_USER_TEMP_DIR, path,
sizeof(path));
508 if (
len >
sizeof(path)) {
509 confstr(_CS_DARWIN_USER_TEMP_DIR, tmpdir,
len);
512 memcpy(tmpdir, path,
len);
521check_tmpdir(
const char *dir)
525 if (!dir)
return FALSE;
526 if (stat(dir, &st))
return FALSE;
528# define S_ISDIR(m) (((m) & S_IFMT) == S_IFDIR)
530 if (!S_ISDIR(st.st_mode))
return FALSE;
535 if (st.st_mode & S_IWOTH) {
537 if (!(st.st_mode & S_ISVTX))
return FALSE;
542 if (access(dir, W_OK))
return FALSE;
551# define RETURN_ENV(name) \
552 if (check_tmpdir(tmpdir = getenv(name))) return ruby_strdup(tmpdir)
553 RETURN_ENV(
"TMPDIR");
555 tmpdir = system_default_tmpdir();
556 if (check_tmpdir(tmpdir))
return tmpdir;
564sprint_ext_filename(
char *str,
size_t size,
long box_id,
const char *prefix,
const char *basename)
566 if (tmp_dir_has_dirsep) {
567 return snprintf(str, size,
"%s%sp%"PRI_PIDT_PREFIX"u_%ld_%s", tmp_dir, prefix, getpid(), box_id, basename);
569 return snprintf(str, size,
"%s%s%sp%"PRI_PIDT_PREFIX"u_%ld_%s", tmp_dir, DIRSEP, prefix, getpid(), box_id, basename);
572enum copy_error_type {
577 COPY_ERROR_DST_WRITE,
579 COPY_ERROR_DST_CHMOD,
584copy_ext_file_error(
char *message,
size_t size,
int copy_retvalue)
587 int error = GetLastError();
589 size_t len = snprintf(message, size,
"%d: ", error);
591#define format_message(sublang) FormatMessage(\
592 FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, \
593 NULL, error, MAKELANGID(LANG_NEUTRAL, (sublang)), \
594 message + len, size - len, NULL)
595 if (format_message(SUBLANG_ENGLISH_US) == 0)
596 format_message(SUBLANG_DEFAULT);
597 for (p = message +
len; *p; p++) {
598 if (*p ==
'\n' || *p ==
'\r')
602 switch (copy_retvalue) {
603 case COPY_ERROR_SRC_OPEN:
604 strlcpy(message,
"can't open the extension path", size);
606 case COPY_ERROR_DST_OPEN:
607 strlcpy(message,
"can't open the file to write", size);
609 case COPY_ERROR_SRC_READ:
610 strlcpy(message,
"failed to read the extension path", size);
612 case COPY_ERROR_DST_WRITE:
613 strlcpy(message,
"failed to write the extension path", size);
615 case COPY_ERROR_SRC_STAT:
616 strlcpy(message,
"failed to stat the extension path to copy permissions", size);
618 case COPY_ERROR_DST_CHMOD:
619 strlcpy(message,
"failed to set permissions to the copied extension path", size);
621 case COPY_ERROR_SYSERR:
622 strlcpy(message, strerror(
errno), size);
624 case COPY_ERROR_NONE:
626 rb_bug(
"unknown return value of copy_ext_file: %d", copy_retvalue);
633static enum copy_error_type
634copy_stream(
int src_fd,
int dst_fd)
639 while ((rsize = read(src_fd, buffer,
sizeof(buffer))) != 0) {
640 if (rsize < 0)
return COPY_ERROR_SRC_READ;
641 for (
size_t written = 0; written < (size_t)rsize;) {
642 ssize_t wsize = write(dst_fd, buffer+written, rsize-written);
643 if (wsize < 0)
return COPY_ERROR_DST_WRITE;
644 written += (size_t)wsize;
647 return COPY_ERROR_NONE;
651static enum copy_error_type
652copy_ext_file(
const char *src_path,
const char *dst_path)
655 WCHAR *w_src = rb_w32_mbstr_to_wstr(CP_UTF8, src_path, -1, NULL);
656 WCHAR *w_dst = rb_w32_mbstr_to_wstr(CP_UTF8, dst_path, -1, NULL);
657 if (!w_src || !w_dst) {
663 enum copy_error_type rvalue = CopyFileW(w_src, w_dst, TRUE) ?
664 COPY_ERROR_NONE : COPY_ERROR_SYSERR;
670 const int bin = O_BINARY;
675 const int cloexec = O_CLOEXEC;
677 const int cloexec = 0;
679 const int src_fd = open(src_path, O_RDONLY|cloexec|bin);
680 if (src_fd < 0)
return COPY_ERROR_SRC_OPEN;
681 if (!cloexec) rb_maygvl_fd_fix_cloexec(src_fd);
684 if (fstat(src_fd, &src_st)) {
686 return COPY_ERROR_SRC_STAT;
689 const int dst_fd = open(dst_path, O_WRONLY|O_CREAT|O_EXCL|cloexec|bin, S_IRWXU);
692 return COPY_ERROR_DST_OPEN;
694 if (!cloexec) rb_maygvl_fd_fix_cloexec(dst_fd);
696 enum copy_error_type ret = COPY_ERROR_NONE;
698 if (fchmod(dst_fd, src_st.st_mode & 0777)) {
699 ret = COPY_ERROR_DST_CHMOD;
703 const size_t count_max = (SIZE_MAX >> 1) + 1;
706# ifdef HAVE_COPY_FILE_RANGE
708 ssize_t written = copy_file_range(src_fd, NULL, dst_fd, NULL, count_max, 0);
709 if (written == 0)
goto done;
710 if (written < 0)
break;
713# ifdef HAVE_FCOPYFILE
714 if (fcopyfile(src_fd, dst_fd, NULL, COPYFILE_DATA) == 0) {
720 ssize_t written = sendfile(src_fd, dst_fd, NULL count_max);
721 if (written == 0)
goto done;
722 if (written < 0)
break;
725 ret = copy_stream(src_fd, dst_fd);
729 if (dst_fd >= 0) close(dst_fd);
730 if (ret != COPY_ERROR_NONE) unlink(dst_path);
735#if defined __CYGWIN__ || defined DOSISH
736#define isdirsep(x) ((x) == '/' || (x) == '\\')
738#define isdirsep(x) ((x) == '/')
741#define IS_SOEXT(e) (strcmp((e), ".so") == 0 || strcmp((e), ".o") == 0)
742#define IS_DLEXT(e) (strcmp((e), DLEXT) == 0)
745fname_without_suffix(
const char *fname,
char *rvalue,
size_t rsize)
747 size_t len = strlen(fname);
749 for (pos = fname +
len; pos > fname; pos--) {
750 if (IS_SOEXT(pos) || IS_DLEXT(pos)) {
754 if (fname +
len - pos > DLEXT_MAXLEN)
break;
756 if (
len > rsize - 1)
len = rsize - 1;
757 memcpy(rvalue, fname,
len);
762escaped_basename(
const char *path,
const char *fname,
char *rvalue,
size_t rsize)
765 const char *leaf = path, *found;
767 while ((found = strstr(leaf + 1, fname)) != NULL) {
770 strlcpy(rvalue, leaf, rsize);
771 for (pos = rvalue; *pos; pos++) {
772 if (isdirsep(*pos)) {
779box_ext_cleanup_mark(
void *p)
781 rb_gc_mark((
VALUE)p);
785box_ext_cleanup_free(
void *p)
788 unlink(RSTRING_PTR(path));
793 {box_ext_cleanup_mark, box_ext_cleanup_free},
798rb_box_cleanup_local_extension(
VALUE cleanup)
803 if (p) box_ext_cleanup_free(p);
812 HMODULE h = (HMODULE)NUM2PTR(value);
813 WCHAR module_path[MAXPATHLEN];
814 DWORD
len = GetModuleFileNameW(h, module_path, numberof(module_path));
817 if (
len > 0 &&
len < numberof(module_path)) DeleteFileW(module_path);
823cleanup_all_local_extensions(
VALUE libmap)
831 char ext_path[MAXPATHLEN], fname2[MAXPATHLEN], basename[MAXPATHLEN];
833 const char *src_path = RSTRING_PTR(path), *fname_ptr = RSTRING_PTR(fname);
834 rb_box_t *box = rb_get_box_t(box_value);
836 fname_without_suffix(fname_ptr, fname2,
sizeof(fname2));
837 escaped_basename(src_path, fname2, basename,
sizeof(basename));
839 wrote = sprint_ext_filename(ext_path,
sizeof(ext_path), box->box_id, BOX_TMP_PREFIX, basename);
840 if (wrote >= (
int)
sizeof(ext_path)) {
841 rb_bug(
"Extension file path in the box was too long");
845 enum copy_error_type copy_error = copy_ext_file(src_path, ext_path);
848 copy_ext_file_error(message,
sizeof(message), copy_error);
849 rb_raise(
rb_eLoadError,
"can't prepare the extension file for Ruby Box (%s from %"PRIsVALUE
"): %s", ext_path, path, message);
851 DATA_PTR(*cleanup) = (
void *)new_path;
861 rb_vm_frame_flag_set_box_require(GET_EC());
863 VALUE args = rb_ary_new_from_args(2, fname, wrap);
864 return rb_load_entrypoint(args);
870 rb_vm_frame_flag_set_box_require(GET_EC());
876rb_box_require_relative(
VALUE box,
VALUE fname)
878 rb_vm_frame_flag_set_box_require(GET_EC());
880 return rb_require_relative_entrypoint(fname);
884initialize_master_box(
void)
892 master->load_path_check_cache = 0;
897 master->loaded_features_index = st_init_numtable();
898 master->loaded_features_realpaths = rb_hash_new();
900 master->loaded_features_realpath_map = rb_hash_new();
903 master->ruby_dln_libmap = rb_hash_new_with_size(0);
904 master->gvar_tbl = rb_hash_new_with_size(0);
905 master->classext_cow_classes = NULL;
907 vm->master_box = master;
909 if (rb_box_available()) {
910 VALUE master_box, entry;
912 CONST_ID(id_box_entry,
"__box_entry__");
915 RCLASS_SET_PRIME_CLASSEXT_WRITABLE(master_box,
true);
916 RCLASS_SET_CONST_TBL(master_box, RCLASSEXT_CONST_TBL(RCLASS_EXT_PRIME(
rb_cObject)),
true);
918 master->box_id = box_generate_id();
919 master->box_object = master_box;
924 rb_gc_register_mark_object(master_box);
925 rb_gc_register_mark_object(entry);
929 master->box_object =
Qnil;
944 box = (
const rb_box_t *)rb_get_box_t(box_value);
946 return rb_iseq_eval(iseq, box);
949static int box_experimental_warned = 0;
954box_value_initialize(
bool root,
bool user,
bool optional)
959 VM_ASSERT(BOX_OBJ_P(box_value));
961 box = rb_get_box_t(box_value);
962 box->box_object = box_value;
965 box->is_optional = optional;
970rb_initialize_mandatory_boxes(
void)
972 VALUE root_box_value, main_box_value;
975 VM_ASSERT(rb_box_available());
977 if (!box_experimental_warned) {
979 "Ruby::Box is experimental, and the behavior may change in the future!\n"
980 "See https://docs.ruby-lang.org/en/%s/Ruby/Box.html for known issues, etc.",
981 ruby_api_version_name);
982 box_experimental_warned = 1;
985 root_box_value = box_value_initialize(
true,
false,
false);
986 main_box_value = box_value_initialize(
false,
true,
false);
991 vm->root_box = root_box = rb_get_box_t(root_box_value);
992 vm->main_box = main_box = rb_get_box_t(main_box_value);
995 RCLASS_EXT_WRITABLE_IN_BOX(
rb_cObject, root_box);
996 RCLASS_EXT_WRITABLE_IN_BOX(
rb_cObject, main_box);
1000rb_box_inspect(
VALUE obj)
1008 box = rb_get_box_t(obj);
1011 if (BOX_MASTER_P(box)) {
1014 if (BOX_ROOT_P(box)) {
1017 if (BOX_USER_P(box)) {
1020 if (BOX_MAIN_P(box)) {
1023 else if (BOX_OPTIONAL_P(box)) {
1031rb_box_loading_func(
int argc,
VALUE *argv,
VALUE _self)
1033 rb_vm_frame_flag_set_box_require(GET_EC());
1038box_define_loader_method(
const char *name)
1045Init_master_box(
void)
1047 master_box->loading_table = st_init_strtable();
1051Init_enable_box(
void)
1053 const char *env = getenv(
"RUBY_BOX");
1054 if (env && strlen(env) == 1 && env[0] ==
'1') {
1055 ruby_box_enabled =
true;
1058 ruby_box_init_done =
true;
1064rb_box_s_master(
VALUE recv)
1066 return master_box->box_object;
1071rb_box_s_root(
VALUE recv)
1073 return root_box->box_object;
1078rb_box_s_main(
VALUE recv)
1080 return main_box->box_object;
1085rb_box_master_p(
VALUE box_value)
1088 return RBOOL(BOX_MASTER_P(box));
1093rb_box_root_p(
VALUE box_value)
1096 return RBOOL(BOX_ROOT_P(box));
1101rb_box_main_p(
VALUE box_value)
1104 return RBOOL(BOX_MAIN_P(box));
1110classname(
VALUE klass)
1116 p = RCLASSEXT_CLASSPATH(RCLASS_EXT_PRIME(klass));
1118 return RSTRING_PTR(p);
1120 return "AnyClassValue";
1121 return "NonClassValue";
1124static enum rb_id_table_iterator_result
1125dump_classext_methods_i(
ID mid,
VALUE _val,
void *data)
1129 return ID_TABLE_CONTINUE;
1132static enum rb_id_table_iterator_result
1133dump_classext_constants_i(
ID mid,
VALUE _val,
void *data)
1137 return ID_TABLE_CONTINUE;
1147 snprintf(buf, 4096,
"Ruby::Box %ld:%s classext %p\n",
1148 RCLASSEXT_BOX(ext)->box_id, is_prime ?
" prime" :
"", (void *)ext);
1151 snprintf(buf, 2048,
" Super: %s\n", classname(RCLASSEXT_SUPER(ext)));
1154 tbl = RCLASSEXT_M_TBL(ext);
1157 rb_id_table_foreach(RCLASSEXT_M_TBL(ext), dump_classext_methods_i, (
void *)ary);
1159 snprintf(buf, 4096,
" Methods(%ld): ",
RARRAY_LEN(ary));
1168 tbl = RCLASSEXT_CONST_TBL(ext);
1171 rb_id_table_foreach(tbl, dump_classext_constants_i, (
void *)ary);
1173 snprintf(buf, 4096,
" Constants(%ld): ",
RARRAY_LEN(ary));
1208 snprintf(buf, 2048,
"Non-class/module value: %p (%s)\n", (
void *)klass, rb_type_str(
BUILTIN_TYPE(klass)));
1213 snprintf(buf, 2048,
"Class: %p (%s)%s\n",
1214 (
void *)klass, classname(klass), RCLASS_SINGLETON_P(klass) ?
" [singleton]" :
"");
1217 snprintf(buf, 2048,
"Module: %p (%s)\n", (
void *)klass, classname(klass));
1221 ext = RCLASS_EXT_PRIME(klass);
1222 box = RCLASSEXT_BOX(ext);
1223 snprintf(buf, 2048,
"Prime classext box(%ld,%s), readable(%s), writable(%s)\n",
1225 BOX_MASTER_P(box) ?
"master" : (BOX_ROOT_P(box) ?
"root" : (BOX_MAIN_P(box) ?
"main" :
"optional")),
1226 RCLASS_PRIME_CLASSEXT_READABLE_P(klass) ?
"t" :
"f",
1227 RCLASS_PRIME_CLASSEXT_WRITABLE_P(klass) ?
"t" :
"f");
1230 classext_tbl = RCLASS_CLASSEXT_TBL(klass);
1231 if (!classext_tbl) {
1235 snprintf(buf, 2048,
"Non-prime classexts: %zu\n", st_table_size(classext_tbl));
1239 rb_class_classext_foreach(klass, dump_classext_i, (
void *)res);
1255 tmp_dir = system_tmpdir();
1256 tmp_dir_has_dirsep = (strcmp(tmp_dir + (strlen(tmp_dir) - strlen(DIRSEP)), DIRSEP) == 0);
1267 initialize_master_box();
1271 box_define_loader_method(
"require");
1272 box_define_loader_method(
"require_relative");
1273 box_define_loader_method(
"load");
1275 if (rb_box_available()) {
#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.
#define rb_define_global_function(mid, func, arity)
Defines rb_mKernel #mid.
#define RUBY_EXTERN
Declaration of externally visible global variables.
Ruby-level global variables / constants, visible from C.
void rb_include_module(VALUE klass, VALUE module)
Includes a module to a class.
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.
VALUE rb_define_module(const char *name)
Defines a top-level module.
VALUE rb_define_module_under(VALUE outer, const char *name)
Defines a module 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_scan_args(int argc, const VALUE *argv, const char *fmt,...)
Retrieves argument from argc and argv to given VALUE references according to the format string.
#define rb_str_new2
Old name of rb_str_new_cstr.
#define xfree
Old name of ruby_xfree.
#define xmalloc
Old name of ruby_xmalloc.
#define T_MODULE
Old name of RUBY_T_MODULE.
#define T_ICLASS
Old name of RUBY_T_ICLASS.
#define LONG2NUM
Old name of RB_LONG2NUM.
#define Qnil
Old name of RUBY_Qnil.
#define Qfalse
Old name of RUBY_Qfalse.
#define NIL_P
Old name of RB_NIL_P.
#define T_CLASS
Old name of RUBY_T_CLASS.
#define BUILTIN_TYPE
Old name of RB_BUILTIN_TYPE.
#define FIXNUM_P
Old name of RB_FIXNUM_P.
#define CONST_ID
Old name of RUBY_CONST_ID.
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_eLoadError
LoadError exception.
VALUE rb_eRuntimeError
RuntimeError exception.
@ RB_WARN_CATEGORY_EXPERIMENTAL
Warning is for experimental features.
VALUE rb_cObject
Object class.
VALUE rb_obj_alloc(VALUE klass)
Allocates an instance of the given class.
VALUE rb_class_new_instance(int argc, const VALUE *argv, VALUE klass)
Allocates, then initialises an instance of the given class.
VALUE rb_cBox
Ruby::Box class.
VALUE rb_obj_hide(VALUE obj)
Make the object invisible from Ruby code.
VALUE rb_class_new_instance_pass_kw(int argc, const VALUE *argv, VALUE klass)
Identical to rb_class_new_instance(), except it passes the passed keywords if any to the #initialize ...
VALUE rb_cModule
Module class.
VALUE rb_funcall(VALUE recv, ID mid, int n,...)
Calls a method.
VALUE rb_call_super(int argc, const VALUE *argv)
This resembles ruby's super.
VALUE rb_ary_dup(VALUE ary)
Duplicates an array.
VALUE rb_ary_new(void)
Allocates a new, empty array.
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_hidden_new(long capa)
Allocates a hidden (no class) empty array.
VALUE rb_ary_push(VALUE ary, VALUE elem)
Special case of rb_ary_cat() that it adds only one element.
VALUE rb_ary_sort_bang(VALUE ary)
Destructively sorts the passed array in-place, according to each elements' <=> result.
VALUE rb_ary_join(VALUE ary, VALUE sep)
Recursively stringises the elements of the passed array, flattens that result, then joins the sequenc...
VALUE rb_require_string(VALUE feature)
Finds and loads the given feature, if absent.
VALUE rb_str_concat(VALUE dst, VALUE src)
Identical to rb_str_append(), except it also accepts an integer as a codepoint.
#define rb_str_cat_cstr(buf, str)
Identical to rb_str_cat(), except it assumes the passed pointer is a pointer to a C string.
#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_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.
void rb_const_set(VALUE space, ID name, VALUE val)
Names a constant.
void rb_define_alloc_func(VALUE klass, rb_alloc_func_t func)
Sets the allocator function of a class.
int len
Length of the buffer.
char * ruby_strdup(const char *str)
This is our own version of strdup(3) that uses ruby_xmalloc() instead of system malloc (benefits our ...
void rb_hash_foreach(VALUE q, int_type *w, VALUE e)
Iteration over the given hash.
#define PRI_PIDT_PREFIX
A rb_sprintf() format prefix to be used for a pid_t parameter.
#define RARRAY_LEN
Just another name of rb_array_len.
#define DATA_PTR(obj)
Convenient getter macro.
#define StringValue(v)
Ensures that the parameter object is a String.
#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_Wrap_Struct(klass, data_type, sval)
Converts sval, a pointer to your struct, into a 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 RTEST
This is an old name of RB_TEST.
Internal header for Ruby Box.
This is the struct that holds necessary info for a struct.
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.