00001
00002
00003
00004
00005
00006
00007
00008
00009
00010
00011
00012 #ifndef RUBY_VM_CORE_H
00013 #define RUBY_VM_CORE_H
00014
00015 #define RUBY_VM_THREAD_MODEL 2
00016
00017 #include "ruby/ruby.h"
00018 #include "ruby/st.h"
00019
00020 #include "node.h"
00021 #include "vm_debug.h"
00022 #include "vm_opts.h"
00023 #include "id.h"
00024 #include "method.h"
00025 #include "ruby_atomic.h"
00026
00027 #if defined(_WIN32)
00028 #include "thread_win32.h"
00029 #elif defined(HAVE_PTHREAD_H)
00030 #include "thread_pthread.h"
00031 #else
00032 #error "unsupported thread type"
00033 #endif
00034
00035 #ifndef ENABLE_VM_OBJSPACE
00036 #ifdef _WIN32
00037
00038
00039
00040
00041
00042
00043
00044 #define ENABLE_VM_OBJSPACE 0
00045 #else
00046 #define ENABLE_VM_OBJSPACE 1
00047 #endif
00048 #endif
00049
00050 #include <setjmp.h>
00051 #include <signal.h>
00052
00053 #ifndef NSIG
00054 # define NSIG (_SIGMAX + 1)
00055 #endif
00056
00057 #define RUBY_NSIG NSIG
00058
00059 #ifdef HAVE_STDARG_PROTOTYPES
00060 #include <stdarg.h>
00061 #define va_init_list(a,b) va_start((a),(b))
00062 #else
00063 #include <varargs.h>
00064 #define va_init_list(a,b) va_start((a))
00065 #endif
00066
00067 #if defined(SIGSEGV) && defined(HAVE_SIGALTSTACK) && defined(SA_SIGINFO) && !defined(__NetBSD__)
00068 #define USE_SIGALTSTACK
00069 #endif
00070
00071
00072
00073
00074
00075
00076 #if defined(__GNUC__) && __GNUC__ >= 2
00077
00078 #if OPT_TOKEN_THREADED_CODE
00079 #if OPT_DIRECT_THREADED_CODE
00080 #undef OPT_DIRECT_THREADED_CODE
00081 #endif
00082 #endif
00083
00084 #else
00085
00086
00087 #if OPT_DIRECT_THREADED_CODE
00088 #undef OPT_DIRECT_THREADED_CODE
00089 #endif
00090 #if OPT_TOKEN_THREADED_CODE
00091 #undef OPT_TOKEN_THREADED_CODE
00092 #endif
00093 #endif
00094
00095 #ifdef __native_client__
00096 #undef OPT_DIRECT_THREADED_CODE
00097 #endif
00098
00099
00100 #if OPT_CALL_THREADED_CODE
00101 #if OPT_DIRECT_THREADED_CODE
00102 #undef OPT_DIRECT_THREADED_CODE
00103 #endif
00104 #if OPT_STACK_CACHING
00105 #undef OPT_STACK_CACHING
00106 #endif
00107 #endif
00108
00109
00110 #if __GNUC__ >= 3
00111 #define LIKELY(x) (__builtin_expect((x), 1))
00112 #define UNLIKELY(x) (__builtin_expect((x), 0))
00113 #else
00114 #define LIKELY(x) (x)
00115 #define UNLIKELY(x) (x)
00116 #endif
00117
00118 #if __GNUC__ >= 3
00119 #define UNINITIALIZED_VAR(x) x = x
00120 #else
00121 #define UNINITIALIZED_VAR(x) x
00122 #endif
00123
00124 typedef unsigned long rb_num_t;
00125
00126
00127
00128 struct iseq_compile_data_ensure_node_stack;
00129
00130 typedef struct rb_compile_option_struct rb_compile_option_t;
00131
00132 struct iseq_inline_cache_entry {
00133 VALUE ic_vmstat;
00134 VALUE ic_class;
00135 union {
00136 VALUE value;
00137 long index;
00138 } ic_value;
00139 };
00140
00141
00142 struct rb_thread_struct;
00143 struct rb_control_frame_struct;
00144
00145
00146 typedef struct rb_call_info_struct {
00147
00148 ID mid;
00149 VALUE flag;
00150 int orig_argc;
00151 rb_iseq_t *blockiseq;
00152
00153
00154 VALUE vmstat;
00155 VALUE klass;
00156
00157
00158 const rb_method_entry_t *me;
00159 VALUE defined_class;
00160
00161
00162 int argc;
00163 struct rb_block_struct *blockptr;
00164 VALUE recv;
00165 union {
00166 int opt_pc;
00167 long index;
00168 int missing_reason;
00169 int inc_sp;
00170 } aux;
00171
00172 VALUE (*call)(struct rb_thread_struct *th, struct rb_control_frame_struct *cfp, struct rb_call_info_struct *ci);
00173 } rb_call_info_t;
00174
00175 #if 1
00176 #define GetCoreDataFromValue(obj, type, ptr) do { \
00177 (ptr) = (type*)DATA_PTR(obj); \
00178 } while (0)
00179 #else
00180 #define GetCoreDataFromValue(obj, type, ptr) Data_Get_Struct((obj), type, (ptr))
00181 #endif
00182
00183 #define GetISeqPtr(obj, ptr) \
00184 GetCoreDataFromValue((obj), rb_iseq_t, (ptr))
00185
00186 typedef struct rb_iseq_location_struct {
00187 VALUE path;
00188 VALUE absolute_path;
00189 VALUE base_label;
00190 VALUE label;
00191 size_t first_lineno;
00192 } rb_iseq_location_t;
00193
00194 struct rb_iseq_struct;
00195
00196 struct rb_iseq_struct {
00197
00198
00199
00200
00201 enum iseq_type {
00202 ISEQ_TYPE_TOP,
00203 ISEQ_TYPE_METHOD,
00204 ISEQ_TYPE_BLOCK,
00205 ISEQ_TYPE_CLASS,
00206 ISEQ_TYPE_RESCUE,
00207 ISEQ_TYPE_ENSURE,
00208 ISEQ_TYPE_EVAL,
00209 ISEQ_TYPE_MAIN,
00210 ISEQ_TYPE_DEFINED_GUARD
00211 } type;
00212
00213 rb_iseq_location_t location;
00214
00215 VALUE *iseq;
00216 VALUE *iseq_encoded;
00217 unsigned long iseq_size;
00218 VALUE mark_ary;
00219 VALUE coverage;
00220
00221
00222 struct iseq_line_info_entry *line_info_table;
00223 size_t line_info_size;
00224
00225 ID *local_table;
00226 int local_table_size;
00227
00228
00229 int local_size;
00230
00231 struct iseq_inline_cache_entry *ic_entries;
00232 int ic_size;
00233
00234 rb_call_info_t *callinfo_entries;
00235 int callinfo_size;
00236
00264 int argc;
00265 int arg_simple;
00266 int arg_rest;
00267 int arg_block;
00268 int arg_opts;
00269 int arg_post_len;
00270 int arg_post_start;
00271 int arg_size;
00272 VALUE *arg_opt_table;
00273 int arg_keyword;
00274 int arg_keyword_check;
00275 int arg_keywords;
00276 ID *arg_keyword_table;
00277
00278 size_t stack_max;
00279
00280
00281 struct iseq_catch_table_entry *catch_table;
00282 int catch_table_size;
00283
00284
00285 struct rb_iseq_struct *parent_iseq;
00286 struct rb_iseq_struct *local_iseq;
00287
00288
00289
00290
00291
00292 VALUE self;
00293 VALUE orig;
00294
00295
00296
00297
00298
00299
00300
00301
00302
00303
00304 NODE *cref_stack;
00305 VALUE klass;
00306
00307
00308 ID defined_method_id;
00309 rb_num_t flip_cnt;
00310
00311
00312 struct iseq_compile_data *compile_data;
00313 };
00314
00315 enum ruby_special_exceptions {
00316 ruby_error_reenter,
00317 ruby_error_nomemory,
00318 ruby_error_sysstack,
00319 ruby_error_closed_stream,
00320 ruby_special_error_count
00321 };
00322
00323 #define GetVMPtr(obj, ptr) \
00324 GetCoreDataFromValue((obj), rb_vm_t, (ptr))
00325
00326 #if defined(ENABLE_VM_OBJSPACE) && ENABLE_VM_OBJSPACE
00327 struct rb_objspace;
00328 void rb_objspace_free(struct rb_objspace *);
00329 #endif
00330
00331 typedef struct rb_hook_list_struct {
00332 struct rb_event_hook_struct *hooks;
00333 rb_event_flag_t events;
00334 int need_clean;
00335 } rb_hook_list_t;
00336
00337 typedef struct rb_vm_struct {
00338 VALUE self;
00339
00340 rb_global_vm_lock_t gvl;
00341 rb_thread_lock_t thread_destruct_lock;
00342
00343 struct rb_thread_struct *main_thread;
00344 struct rb_thread_struct *running_thread;
00345
00346 st_table *living_threads;
00347 VALUE thgroup_default;
00348
00349 int running;
00350 int thread_abort_on_exception;
00351 int trace_running;
00352 volatile int sleeper;
00353
00354
00355 VALUE mark_object_ary;
00356
00357 VALUE special_exceptions[ruby_special_error_count];
00358
00359
00360 VALUE top_self;
00361 VALUE load_path;
00362 VALUE load_path_snapshot;
00363 VALUE load_path_check_cache;
00364 VALUE expanded_load_path;
00365 VALUE loaded_features;
00366 VALUE loaded_features_snapshot;
00367 struct st_table *loaded_features_index;
00368 struct st_table *loading_table;
00369
00370
00371 struct {
00372 VALUE cmd;
00373 int safe;
00374 } trap_list[RUBY_NSIG];
00375
00376
00377 rb_hook_list_t event_hooks;
00378
00379 int src_encoding_index;
00380
00381 VALUE verbose, debug, progname;
00382 VALUE coverages;
00383
00384 struct unlinked_method_entry_list_entry *unlinked_method_entry_list;
00385
00386 #if defined(ENABLE_VM_OBJSPACE) && ENABLE_VM_OBJSPACE
00387 struct rb_objspace *objspace;
00388 #endif
00389
00390
00391
00392
00393
00394 struct RArray at_exit;
00395
00396 VALUE *defined_strings;
00397
00398
00399 struct {
00400 size_t thread_vm_stack_size;
00401 size_t thread_machine_stack_size;
00402 size_t fiber_vm_stack_size;
00403 size_t fiber_machine_stack_size;
00404 } default_params;
00405 } rb_vm_t;
00406
00407
00408
00409 #define RUBY_VM_SIZE_ALIGN 4096
00410
00411 #define RUBY_VM_THREAD_VM_STACK_SIZE ( 128 * 1024 * sizeof(VALUE))
00412 #define RUBY_VM_THREAD_VM_STACK_SIZE_MIN ( 2 * 1024 * sizeof(VALUE))
00413 #define RUBY_VM_THREAD_MACHINE_STACK_SIZE ( 128 * 1024 * sizeof(VALUE))
00414 #define RUBY_VM_THREAD_MACHINE_STACK_SIZE_MIN ( 16 * 1024 * sizeof(VALUE))
00415
00416 #define RUBY_VM_FIBER_VM_STACK_SIZE ( 16 * 1024 * sizeof(VALUE))
00417 #define RUBY_VM_FIBER_VM_STACK_SIZE_MIN ( 2 * 1024 * sizeof(VALUE))
00418 #define RUBY_VM_FIBER_MACHINE_STACK_SIZE ( 64 * 1024 * sizeof(VALUE))
00419 #define RUBY_VM_FIBER_MACHINE_STACK_SIZE_MIN ( 16 * 1024 * sizeof(VALUE))
00420
00421 #ifndef VM_DEBUG_BP_CHECK
00422 #define VM_DEBUG_BP_CHECK 0
00423 #endif
00424
00425 typedef struct rb_control_frame_struct {
00426 VALUE *pc;
00427 VALUE *sp;
00428 rb_iseq_t *iseq;
00429 VALUE flag;
00430 VALUE self;
00431 VALUE klass;
00432 VALUE *ep;
00433 rb_iseq_t *block_iseq;
00434 VALUE proc;
00435 const rb_method_entry_t *me;
00436
00437 #if VM_DEBUG_BP_CHECK
00438 VALUE *bp_check;
00439 #endif
00440 } rb_control_frame_t;
00441
00442 typedef struct rb_block_struct {
00443 VALUE self;
00444 VALUE klass;
00445 VALUE *ep;
00446 rb_iseq_t *iseq;
00447 VALUE proc;
00448 } rb_block_t;
00449
00450 extern const rb_data_type_t ruby_threadptr_data_type;
00451
00452 #define GetThreadPtr(obj, ptr) \
00453 TypedData_Get_Struct((obj), rb_thread_t, &ruby_threadptr_data_type, (ptr))
00454
00455 enum rb_thread_status {
00456 THREAD_RUNNABLE,
00457 THREAD_STOPPED,
00458 THREAD_STOPPED_FOREVER,
00459 THREAD_KILLED
00460 };
00461
00462 typedef RUBY_JMP_BUF rb_jmpbuf_t;
00463
00464
00465
00466
00467
00468 struct rb_vm_tag {
00469 VALUE tag;
00470 VALUE retval;
00471 rb_jmpbuf_t buf;
00472 struct rb_vm_tag *prev;
00473 };
00474
00475 struct rb_vm_protect_tag {
00476 struct rb_vm_protect_tag *prev;
00477 };
00478
00479 struct rb_unblock_callback {
00480 rb_unblock_function_t *func;
00481 void *arg;
00482 };
00483
00484 struct rb_mutex_struct;
00485
00486 struct rb_thread_struct;
00487 typedef struct rb_thread_list_struct{
00488 struct rb_thread_list_struct *next;
00489 struct rb_thread_struct *th;
00490 } rb_thread_list_t;
00491
00492
00493 typedef struct rb_thread_struct {
00494 VALUE self;
00495 rb_vm_t *vm;
00496
00497
00498 VALUE *stack;
00499 size_t stack_size;
00500 rb_control_frame_t *cfp;
00501 int safe_level;
00502 int raised_flag;
00503 VALUE last_status;
00504
00505
00506 int state;
00507
00508 int waiting_fd;
00509
00510
00511 const rb_block_t *passed_block;
00512
00513
00514 const rb_method_entry_t *passed_me;
00515
00516
00517 rb_call_info_t *passed_ci;
00518
00519
00520 VALUE top_self;
00521 VALUE top_wrapper;
00522
00523
00524 rb_block_t *base_block;
00525
00526 VALUE *root_lep;
00527 VALUE root_svar;
00528
00529
00530 rb_thread_id_t thread_id;
00531 enum rb_thread_status status;
00532 int to_kill;
00533 int priority;
00534
00535 native_thread_data_t native_thread_data;
00536 void *blocking_region_buffer;
00537
00538 VALUE thgroup;
00539 VALUE value;
00540
00541
00542 VALUE errinfo;
00543
00544
00545 #if OPT_CALL_THREADED_CODE
00546 VALUE retval;
00547 #endif
00548
00549
00550 VALUE pending_interrupt_queue;
00551 int pending_interrupt_queue_checked;
00552 VALUE pending_interrupt_mask_stack;
00553
00554 rb_atomic_t interrupt_flag;
00555 unsigned long interrupt_mask;
00556 rb_thread_lock_t interrupt_lock;
00557 struct rb_unblock_callback unblock;
00558 VALUE locking_mutex;
00559 struct rb_mutex_struct *keeping_mutexes;
00560
00561 struct rb_vm_tag *tag;
00562 struct rb_vm_protect_tag *protect_tag;
00563
00570 int parse_in_eval;
00571
00576 int mild_compile_error;
00577
00578
00579 st_table *local_storage;
00580
00581 rb_thread_list_t *join_list;
00582
00583 VALUE first_proc;
00584 VALUE first_args;
00585 VALUE (*first_func)(ANYARGS);
00586
00587
00588 VALUE *machine_stack_start;
00589 VALUE *machine_stack_end;
00590 size_t machine_stack_maxsize;
00591 #ifdef __ia64
00592 VALUE *machine_register_stack_start;
00593 VALUE *machine_register_stack_end;
00594 size_t machine_register_stack_maxsize;
00595 #endif
00596 jmp_buf machine_regs;
00597 int mark_stack_len;
00598
00599
00600 VALUE stat_insn_usage;
00601
00602
00603 rb_hook_list_t event_hooks;
00604 struct rb_trace_arg_struct *trace_arg;
00605
00606
00607 VALUE fiber;
00608 VALUE root_fiber;
00609 rb_jmpbuf_t root_jmpbuf;
00610
00611
00612 int method_missing_reason;
00613 int abort_on_exception;
00614 #ifdef USE_SIGALTSTACK
00615 void *altstack;
00616 #endif
00617 unsigned long running_time_us;
00618 } rb_thread_t;
00619
00620 typedef enum {
00621 VM_DEFINECLASS_TYPE_CLASS = 0x00,
00622 VM_DEFINECLASS_TYPE_SINGLETON_CLASS = 0x01,
00623 VM_DEFINECLASS_TYPE_MODULE = 0x02,
00624
00625 VM_DEFINECLASS_TYPE_MASK = 0x07,
00626 } rb_vm_defineclass_type_t;
00627
00628 #define VM_DEFINECLASS_TYPE(x) ((rb_vm_defineclass_type_t)(x) & VM_DEFINECLASS_TYPE_MASK)
00629 #define VM_DEFINECLASS_FLAG_SCOPED 0x08
00630 #define VM_DEFINECLASS_FLAG_HAS_SUPERCLASS 0x10
00631 #define VM_DEFINECLASS_SCOPED_P(x) ((x) & VM_DEFINECLASS_FLAG_SCOPED)
00632 #define VM_DEFINECLASS_HAS_SUPERCLASS_P(x) \
00633 ((x) & VM_DEFINECLASS_FLAG_HAS_SUPERCLASS)
00634
00635
00636 #if defined __GNUC__ && __GNUC__ >= 4
00637 #pragma GCC visibility push(default)
00638 #endif
00639
00640
00641 VALUE rb_iseq_new(NODE*, VALUE, VALUE, VALUE, VALUE, enum iseq_type);
00642 VALUE rb_iseq_new_top(NODE *node, VALUE name, VALUE path, VALUE absolute_path, VALUE parent);
00643 VALUE rb_iseq_new_main(NODE *node, VALUE path, VALUE absolute_path);
00644 VALUE rb_iseq_new_with_bopt(NODE*, VALUE, VALUE, VALUE, VALUE, VALUE, enum iseq_type, VALUE);
00645 VALUE rb_iseq_new_with_opt(NODE*, VALUE, VALUE, VALUE, VALUE, VALUE, enum iseq_type, const rb_compile_option_t*);
00646
00647
00648 VALUE rb_iseq_compile(VALUE src, VALUE file, VALUE line);
00649 VALUE rb_iseq_compile_on_base(VALUE src, VALUE file, VALUE line, rb_block_t *base_block);
00650 VALUE rb_iseq_compile_with_option(VALUE src, VALUE file, VALUE absolute_path, VALUE line, rb_block_t *base_block, VALUE opt);
00651
00652 VALUE rb_iseq_disasm(VALUE self);
00653 int rb_iseq_disasm_insn(VALUE str, VALUE *iseqval, size_t pos, rb_iseq_t *iseq, VALUE child);
00654 const char *ruby_node_name(int node);
00655 int rb_iseq_first_lineno(const rb_iseq_t *iseq);
00656
00657 RUBY_EXTERN VALUE rb_cISeq;
00658 RUBY_EXTERN VALUE rb_cRubyVM;
00659 RUBY_EXTERN VALUE rb_cEnv;
00660 RUBY_EXTERN VALUE rb_mRubyVMFrozenCore;
00661 #if defined __GNUC__ && __GNUC__ >= 4
00662 #pragma GCC visibility pop
00663 #endif
00664
00665 #define GetProcPtr(obj, ptr) \
00666 GetCoreDataFromValue((obj), rb_proc_t, (ptr))
00667
00668 typedef struct {
00669 rb_block_t block;
00670
00671 VALUE envval;
00672 VALUE blockprocval;
00673 int safe_level;
00674 int is_from_method;
00675 int is_lambda;
00676 } rb_proc_t;
00677
00678 #define GetEnvPtr(obj, ptr) \
00679 GetCoreDataFromValue((obj), rb_env_t, (ptr))
00680
00681 typedef struct {
00682 VALUE *env;
00683 int env_size;
00684 int local_size;
00685 VALUE prev_envval;
00686 rb_block_t block;
00687 } rb_env_t;
00688
00689 #define GetBindingPtr(obj, ptr) \
00690 GetCoreDataFromValue((obj), rb_binding_t, (ptr))
00691
00692 typedef struct {
00693 VALUE env;
00694 VALUE path;
00695 VALUE blockprocval;
00696 unsigned short first_lineno;
00697 } rb_binding_t;
00698
00699
00700
00701 enum vm_check_match_type {
00702 VM_CHECKMATCH_TYPE_WHEN = 1,
00703 VM_CHECKMATCH_TYPE_CASE = 2,
00704 VM_CHECKMATCH_TYPE_RESCUE = 3
00705 };
00706
00707 #define VM_CHECKMATCH_TYPE_MASK 0x03
00708 #define VM_CHECKMATCH_ARRAY 0x04
00709
00710 #define VM_CALL_ARGS_SPLAT (0x01 << 1)
00711 #define VM_CALL_ARGS_BLOCKARG (0x01 << 2)
00712 #define VM_CALL_FCALL (0x01 << 3)
00713 #define VM_CALL_VCALL (0x01 << 4)
00714 #define VM_CALL_TAILCALL (0x01 << 5)
00715 #define VM_CALL_SUPER (0x01 << 6)
00716 #define VM_CALL_OPT_SEND (0x01 << 7)
00717 #define VM_CALL_ARGS_SKIP_SETUP (0x01 << 8)
00718
00719 enum vm_special_object_type {
00720 VM_SPECIAL_OBJECT_VMCORE = 1,
00721 VM_SPECIAL_OBJECT_CBASE,
00722 VM_SPECIAL_OBJECT_CONST_BASE
00723 };
00724
00725 #define VM_FRAME_MAGIC_METHOD 0x11
00726 #define VM_FRAME_MAGIC_BLOCK 0x21
00727 #define VM_FRAME_MAGIC_CLASS 0x31
00728 #define VM_FRAME_MAGIC_TOP 0x41
00729 #define VM_FRAME_MAGIC_CFUNC 0x61
00730 #define VM_FRAME_MAGIC_PROC 0x71
00731 #define VM_FRAME_MAGIC_IFUNC 0x81
00732 #define VM_FRAME_MAGIC_EVAL 0x91
00733 #define VM_FRAME_MAGIC_LAMBDA 0xa1
00734 #define VM_FRAME_MAGIC_RESCUE 0xb1
00735 #define VM_FRAME_MAGIC_MASK_BITS 8
00736 #define VM_FRAME_MAGIC_MASK (~(~0<<VM_FRAME_MAGIC_MASK_BITS))
00737
00738 #define VM_FRAME_TYPE(cfp) ((cfp)->flag & VM_FRAME_MAGIC_MASK)
00739
00740
00741 #define VM_FRAME_FLAG_PASSED 0x0100
00742 #define VM_FRAME_FLAG_FINISH 0x0200
00743 #define VM_FRAME_TYPE_FINISH_P(cfp) (((cfp)->flag & VM_FRAME_FLAG_FINISH) != 0)
00744
00745 #define RUBYVM_CFUNC_FRAME_P(cfp) \
00746 (VM_FRAME_TYPE(cfp) == VM_FRAME_MAGIC_CFUNC)
00747
00748
00749 typedef struct iseq_inline_cache_entry *IC;
00750 typedef rb_call_info_t *CALL_INFO;
00751
00752 void rb_vm_change_state(void);
00753
00754 typedef VALUE CDHASH;
00755
00756 #ifndef FUNC_FASTCALL
00757 #define FUNC_FASTCALL(x) x
00758 #endif
00759
00760 typedef rb_control_frame_t *
00761 (FUNC_FASTCALL(*rb_insn_func_t))(rb_thread_t *, rb_control_frame_t *);
00762
00763 #define GC_GUARDED_PTR(p) ((VALUE)((VALUE)(p) | 0x01))
00764 #define GC_GUARDED_PTR_REF(p) ((void *)(((VALUE)(p)) & ~0x03))
00765 #define GC_GUARDED_PTR_P(p) (((VALUE)(p)) & 0x01)
00766
00767
00768
00769
00770
00771
00772
00773
00774
00775
00776 #define VM_ENVVAL_BLOCK_PTR_FLAG 0x02
00777 #define VM_ENVVAL_BLOCK_PTR(v) (GC_GUARDED_PTR(v) | VM_ENVVAL_BLOCK_PTR_FLAG)
00778 #define VM_ENVVAL_BLOCK_PTR_P(v) ((v) & VM_ENVVAL_BLOCK_PTR_FLAG)
00779 #define VM_ENVVAL_PREV_EP_PTR(v) ((VALUE)GC_GUARDED_PTR(v))
00780 #define VM_ENVVAL_PREV_EP_PTR_P(v) (!(VM_ENVVAL_BLOCK_PTR_P(v)))
00781
00782 #define VM_EP_PREV_EP(ep) ((VALUE *)GC_GUARDED_PTR_REF((ep)[0]))
00783 #define VM_EP_BLOCK_PTR(ep) ((rb_block_t *)GC_GUARDED_PTR_REF((ep)[0]))
00784 #define VM_EP_LEP_P(ep) VM_ENVVAL_BLOCK_PTR_P((ep)[0])
00785
00786 VALUE *rb_vm_ep_local_ep(VALUE *ep);
00787 rb_block_t *rb_vm_control_frame_block_ptr(rb_control_frame_t *cfp);
00788
00789 #define RUBY_VM_PREVIOUS_CONTROL_FRAME(cfp) ((cfp)+1)
00790 #define RUBY_VM_NEXT_CONTROL_FRAME(cfp) ((cfp)-1)
00791 #define RUBY_VM_END_CONTROL_FRAME(th) \
00792 ((rb_control_frame_t *)((th)->stack + (th)->stack_size))
00793 #define RUBY_VM_VALID_CONTROL_FRAME_P(cfp, ecfp) \
00794 ((void *)(ecfp) > (void *)(cfp))
00795 #define RUBY_VM_CONTROL_FRAME_STACK_OVERFLOW_P(th, cfp) \
00796 (!RUBY_VM_VALID_CONTROL_FRAME_P((cfp), RUBY_VM_END_CONTROL_FRAME(th)))
00797
00798 #define RUBY_VM_IFUNC_P(ptr) (BUILTIN_TYPE(ptr) == T_NODE)
00799 #define RUBY_VM_NORMAL_ISEQ_P(ptr) \
00800 ((ptr) && !RUBY_VM_IFUNC_P(ptr))
00801
00802 #define RUBY_VM_GET_BLOCK_PTR_IN_CFP(cfp) ((rb_block_t *)(&(cfp)->self))
00803 #define RUBY_VM_GET_CFP_FROM_BLOCK_PTR(b) \
00804 ((rb_control_frame_t *)((VALUE *)(b) - 4))
00805
00806
00807
00808 VALUE rb_thread_alloc(VALUE klass);
00809 VALUE rb_proc_alloc(VALUE klass);
00810 VALUE rb_binding_alloc(VALUE klass);
00811
00812
00813 extern void rb_vmdebug_stack_dump_raw(rb_thread_t *, rb_control_frame_t *);
00814 extern void rb_vmdebug_debug_print_pre(rb_thread_t *th, rb_control_frame_t *cfp);
00815 extern void rb_vmdebug_debug_print_post(rb_thread_t *th, rb_control_frame_t *cfp);
00816
00817 #define SDR() rb_vmdebug_stack_dump_raw(GET_THREAD(), GET_THREAD()->cfp)
00818 #define SDR2(cfp) rb_vmdebug_stack_dump_raw(GET_THREAD(), (cfp))
00819 void rb_vm_bugreport(void);
00820
00821
00822 #if defined __GNUC__ && __GNUC__ >= 4
00823 #pragma GCC visibility push(default)
00824 #endif
00825 VALUE rb_iseq_eval(VALUE iseqval);
00826 VALUE rb_iseq_eval_main(VALUE iseqval);
00827 #if defined __GNUC__ && __GNUC__ >= 4
00828 #pragma GCC visibility pop
00829 #endif
00830 int rb_thread_method_id_and_class(rb_thread_t *th, ID *idp, VALUE *klassp);
00831
00832 VALUE rb_vm_invoke_proc(rb_thread_t *th, rb_proc_t *proc,
00833 int argc, const VALUE *argv, const rb_block_t *blockptr);
00834 VALUE rb_vm_make_proc(rb_thread_t *th, const rb_block_t *block, VALUE klass);
00835 VALUE rb_vm_make_binding(rb_thread_t *th, const rb_control_frame_t *src_cfp);
00836 VALUE rb_vm_make_env_object(rb_thread_t *th, rb_control_frame_t *cfp);
00837 VALUE rb_binding_new_with_cfp(rb_thread_t *th, const rb_control_frame_t *src_cfp);
00838 void rb_vm_inc_const_missing_count(void);
00839 void rb_vm_gvl_destroy(rb_vm_t *vm);
00840 VALUE rb_vm_call(rb_thread_t *th, VALUE recv, VALUE id, int argc,
00841 const VALUE *argv, const rb_method_entry_t *me,
00842 VALUE defined_class);
00843 void rb_unlink_method_entry(rb_method_entry_t *me);
00844 void rb_gc_mark_unlinked_live_method_entries(void *pvm);
00845
00846 void rb_thread_start_timer_thread(void);
00847 void rb_thread_stop_timer_thread(int);
00848 void rb_thread_reset_timer_thread(void);
00849 void rb_thread_wakeup_timer_thread(void);
00850
00851 int ruby_thread_has_gvl_p(void);
00852 typedef int rb_backtrace_iter_func(void *, VALUE, int, VALUE);
00853 rb_control_frame_t *rb_vm_get_ruby_level_next_cfp(rb_thread_t *th, const rb_control_frame_t *cfp);
00854 rb_control_frame_t *rb_vm_get_binding_creatable_next_cfp(rb_thread_t *th, const rb_control_frame_t *cfp);
00855 int rb_vm_get_sourceline(const rb_control_frame_t *);
00856 VALUE rb_name_err_mesg_new(VALUE obj, VALUE mesg, VALUE recv, VALUE method);
00857 void rb_vm_stack_to_heap(rb_thread_t *th);
00858 void ruby_thread_init_stack(rb_thread_t *th);
00859 int rb_vm_control_frame_id_and_class(const rb_control_frame_t *cfp, ID *idp, VALUE *klassp);
00860 void rb_vm_rewind_cfp(rb_thread_t *th, rb_control_frame_t *cfp);
00861
00862 void rb_gc_mark_machine_stack(rb_thread_t *th);
00863
00864 int rb_autoloading_value(VALUE mod, ID id, VALUE* value);
00865
00866 void rb_vm_rewrite_cref_stack(NODE *node, VALUE old_klass, VALUE new_klass, NODE **new_cref_ptr);
00867
00868 #define sysstack_error GET_VM()->special_exceptions[ruby_error_sysstack]
00869
00870 #define CHECK_VM_STACK_OVERFLOW(cfp, margin) do \
00871 if ((VALUE *)((char *)(((VALUE *)(cfp)->sp) + (margin)) + sizeof(rb_control_frame_t)) >= ((VALUE *)(cfp))) { \
00872 vm_stackoverflow(); \
00873 } \
00874 while (0)
00875
00876
00877
00878 #if RUBY_VM_THREAD_MODEL == 2
00879 extern rb_thread_t *ruby_current_thread;
00880 extern rb_vm_t *ruby_current_vm;
00881 extern rb_event_flag_t ruby_vm_event_flags;
00882
00883 #define GET_VM() ruby_current_vm
00884
00885 #ifndef OPT_CALL_CFUNC_WITHOUT_FRAME
00886 #define OPT_CALL_CFUNC_WITHOUT_FRAME 0
00887 #endif
00888
00889 static inline rb_thread_t *
00890 GET_THREAD(void)
00891 {
00892 rb_thread_t *th = ruby_current_thread;
00893 #if OPT_CALL_CFUNC_WITHOUT_FRAME
00894 if (UNLIKELY(th->passed_ci != 0)) {
00895 void vm_call_cfunc_push_frame(rb_thread_t *th);
00896 vm_call_cfunc_push_frame(th);
00897 }
00898 #endif
00899 return th;
00900 }
00901
00902 #define rb_thread_set_current_raw(th) (void)(ruby_current_thread = (th))
00903 #define rb_thread_set_current(th) do { \
00904 if ((th)->vm->running_thread != (th)) { \
00905 (th)->running_time_us = 0; \
00906 } \
00907 rb_thread_set_current_raw(th); \
00908 (th)->vm->running_thread = (th); \
00909 } while (0)
00910
00911 #else
00912 #error "unsupported thread model"
00913 #endif
00914
00915 enum {
00916 TIMER_INTERRUPT_MASK = 0x01,
00917 PENDING_INTERRUPT_MASK = 0x02,
00918 FINALIZER_INTERRUPT_MASK = 0x04,
00919 TRAP_INTERRUPT_MASK = 0x08
00920 };
00921
00922 #define RUBY_VM_SET_TIMER_INTERRUPT(th) ATOMIC_OR((th)->interrupt_flag, TIMER_INTERRUPT_MASK)
00923 #define RUBY_VM_SET_INTERRUPT(th) ATOMIC_OR((th)->interrupt_flag, PENDING_INTERRUPT_MASK)
00924 #define RUBY_VM_SET_FINALIZER_INTERRUPT(th) ATOMIC_OR((th)->interrupt_flag, FINALIZER_INTERRUPT_MASK)
00925 #define RUBY_VM_SET_TRAP_INTERRUPT(th) ATOMIC_OR((th)->interrupt_flag, TRAP_INTERRUPT_MASK)
00926 #define RUBY_VM_INTERRUPTED(th) ((th)->interrupt_flag & ~(th)->interrupt_mask & (PENDING_INTERRUPT_MASK|TRAP_INTERRUPT_MASK))
00927 #define RUBY_VM_INTERRUPTED_ANY(th) ((th)->interrupt_flag & ~(th)->interrupt_mask)
00928
00929 int rb_signal_buff_size(void);
00930 void rb_signal_exec(rb_thread_t *th, int sig);
00931 void rb_threadptr_check_signal(rb_thread_t *mth);
00932 void rb_threadptr_signal_raise(rb_thread_t *th, int sig);
00933 void rb_threadptr_signal_exit(rb_thread_t *th);
00934 void rb_threadptr_execute_interrupts(rb_thread_t *, int);
00935 void rb_threadptr_interrupt(rb_thread_t *th);
00936 void rb_threadptr_unlock_all_locking_mutexes(rb_thread_t *th);
00937 void rb_threadptr_pending_interrupt_clear(rb_thread_t *th);
00938 void rb_threadptr_pending_interrupt_enque(rb_thread_t *th, VALUE v);
00939 int rb_threadptr_pending_interrupt_active_p(rb_thread_t *th);
00940
00941 void rb_thread_lock_unlock(rb_thread_lock_t *);
00942 void rb_thread_lock_destroy(rb_thread_lock_t *);
00943
00944 #define RUBY_VM_CHECK_INTS_BLOCKING(th) do { \
00945 if (UNLIKELY(!rb_threadptr_pending_interrupt_empty_p(th))) { \
00946 th->pending_interrupt_queue_checked = 0; \
00947 RUBY_VM_SET_INTERRUPT(th); \
00948 rb_threadptr_execute_interrupts(th, 1); \
00949 } \
00950 else if (UNLIKELY(RUBY_VM_INTERRUPTED_ANY(th))) { \
00951 rb_threadptr_execute_interrupts(th, 1); \
00952 } \
00953 } while (0)
00954
00955 #define RUBY_VM_CHECK_INTS(th) do { \
00956 if (UNLIKELY(RUBY_VM_INTERRUPTED_ANY(th))) { \
00957 rb_threadptr_execute_interrupts(th, 0); \
00958 } \
00959 } while (0)
00960
00961
00962 struct rb_trace_arg_struct {
00963 rb_event_flag_t event;
00964 rb_thread_t *th;
00965 rb_control_frame_t *cfp;
00966 VALUE self;
00967 ID id;
00968 VALUE klass;
00969 VALUE data;
00970
00971 int klass_solved;
00972
00973
00974 int lineno;
00975 VALUE path;
00976 };
00977
00978 void rb_threadptr_exec_event_hooks(struct rb_trace_arg_struct *trace_arg);
00979 void rb_threadptr_exec_event_hooks_and_pop_frame(struct rb_trace_arg_struct *trace_arg);
00980
00981 #define EXEC_EVENT_HOOK_ORIG(th_, flag_, self_, id_, klass_, data_, pop_p_) do { \
00982 if (UNLIKELY(ruby_vm_event_flags & (flag_))) { \
00983 if (((th)->event_hooks.events | (th)->vm->event_hooks.events) & (flag_)) { \
00984 struct rb_trace_arg_struct trace_arg; \
00985 trace_arg.event = (flag_); \
00986 trace_arg.th = (th_); \
00987 trace_arg.cfp = (trace_arg.th)->cfp; \
00988 trace_arg.self = (self_); \
00989 trace_arg.id = (id_); \
00990 trace_arg.klass = (klass_); \
00991 trace_arg.data = (data_); \
00992 trace_arg.path = Qundef; \
00993 trace_arg.klass_solved = 0; \
00994 if (pop_p_) rb_threadptr_exec_event_hooks_and_pop_frame(&trace_arg); \
00995 else rb_threadptr_exec_event_hooks(&trace_arg); \
00996 } \
00997 } \
00998 } while (0)
00999
01000 #define EXEC_EVENT_HOOK(th_, flag_, self_, id_, klass_, data_) \
01001 EXEC_EVENT_HOOK_ORIG(th_, flag_, self_, id_, klass_, data_, 0)
01002
01003 #define EXEC_EVENT_HOOK_AND_POP_FRAME(th_, flag_, self_, id_, klass_, data_) \
01004 EXEC_EVENT_HOOK_ORIG(th_, flag_, self_, id_, klass_, data_, 1)
01005
01006 #if defined __GNUC__ && __GNUC__ >= 4
01007 #pragma GCC visibility push(default)
01008 #endif
01009
01010 int rb_thread_check_trap_pending(void);
01011
01012 extern VALUE rb_get_coverages(void);
01013 extern void rb_set_coverages(VALUE);
01014 extern void rb_reset_coverages(void);
01015
01016 #if defined __GNUC__ && __GNUC__ >= 4
01017 #pragma GCC visibility pop
01018 #endif
01019
01020 #endif
01021