summaryrefslogtreecommitdiffhomepage
path: root/src/vm_core.h
diff options
context:
space:
mode:
Diffstat (limited to 'src/vm_core.h')
-rw-r--r--src/vm_core.h414
1 files changed, 414 insertions, 0 deletions
diff --git a/src/vm_core.h b/src/vm_core.h
new file mode 100644
index 000000000..98da043a2
--- /dev/null
+++ b/src/vm_core.h
@@ -0,0 +1,414 @@
+/**********************************************************************
+
+ vm_core.h -
+
+ $Author: yugui $
+ created at: 04/01/01 19:41:38 JST
+
+ Copyright (C) 2004-2007 Koichi Sasada
+
+**********************************************************************/
+
+#ifndef RUBY_VM_CORE_H
+#define RUBY_VM_CORE_H
+
+#define RUBY_VM_THREAD_MODEL 2
+
+//#include "ruby/ruby.h"
+#include "st.h" /* define ANYARGS */
+
+//#include "node.h"
+//#include "debug.h"
+//#include "vm_opts.h"
+//#include "id.h"
+#include "method.h"
+
+#if defined(_WIN32)
+#include "thread_win32.h"
+#elif defined(HAVE_PTHREAD_H)
+#include "thread_pthread.h"
+#else
+#error "unsupported thread type"
+#endif
+
+#ifndef ENABLE_VM_OBJSPACE
+#ifdef _WIN32
+/*
+ * TODO: object space indenpendent st_table.
+ * socklist needs st_table in mrb_w32_sysinit(), before object space
+ * initialization.
+ * It is too early now to change st_hash_type, since it breaks binary
+ * compatibility.
+ */
+#define ENABLE_VM_OBJSPACE 0
+#else
+#define ENABLE_VM_OBJSPACE 1
+#endif
+#endif
+
+#include <setjmp.h>
+#include <signal.h>
+
+//#ifndef NSIG
+//# define NSIG (_SIGMAX + 1) /* For QNX */
+//#endif
+
+//#define RUBY_NSIG NSIG
+
+#ifdef HAVE_STDARG_PROTOTYPES
+#include <stdarg.h>
+#define va_init_list(a,b) va_start(a,b)
+#else
+#include <varargs.h>
+#define va_init_list(a,b) va_start(a)
+#endif
+
+#if defined(SIGSEGV) && defined(HAVE_SIGALTSTACK) && defined(SA_SIGINFO) && !defined(__NetBSD__)
+#define USE_SIGALTSTACK
+#endif
+
+/*****************/
+/* configuration */
+/*****************/
+
+/* gcc ver. check */
+#if defined(__GNUC__) && __GNUC__ >= 2
+
+#if OPT_TOKEN_THREADED_CODE
+#if OPT_DIRECT_THREADED_CODE
+#undef OPT_DIRECT_THREADED_CODE
+#endif
+#endif
+
+#else /* defined(__GNUC__) && __GNUC__ >= 2 */
+
+/* disable threaded code options */
+#if OPT_DIRECT_THREADED_CODE
+#undef OPT_DIRECT_THREADED_CODE
+#endif
+#if OPT_TOKEN_THREADED_CODE
+#undef OPT_TOKEN_THREADED_CODE
+#endif
+#endif
+
+/* call threaded code */
+#if OPT_CALL_THREADED_CODE
+#if OPT_DIRECT_THREADED_CODE
+#undef OPT_DIRECT_THREADED_CODE
+#endif /* OPT_DIRECT_THREADED_CODE */
+#if OPT_STACK_CACHING
+#undef OPT_STACK_CACHING
+#endif /* OPT_STACK_CACHING */
+#endif /* OPT_CALL_THREADED_CODE */
+
+/* likely */
+#if __GNUC__ >= 3
+#define LIKELY(x) (__builtin_expect((x), 1))
+#define UNLIKELY(x) (__builtin_expect((x), 0))
+#else /* __GNUC__ >= 3 */
+#define LIKELY(x) (x)
+#define UNLIKELY(x) (x)
+#endif /* __GNUC__ >= 3 */
+
+#if __GNUC__ >= 3
+#define UNINITIALIZED_VAR(x) x = x
+#else
+#define UNINITIALIZED_VAR(x) x
+#endif
+
+typedef unsigned long mrb_num_t;
+
+/* iseq data type */
+
+struct iseq_compile_data_ensure_node_stack;
+
+typedef struct mrb_compile_option_struct {
+ int inline_const_cache;
+ int peephole_optimization;
+ int tailcall_optimization;
+ int specialized_instruction;
+ int operands_unification;
+ int instructions_unification;
+ int stack_caching;
+ int trace_instruction;
+ int debug_level;
+} mrb_compile_option_t;
+
+struct iseq_inline_cache_entry {
+ mrb_value ic_vmstat;
+ mrb_value ic_class;
+ union {
+ mrb_value value;
+ mrb_method_entry_t *method;
+ long index;
+ } ic_value;
+};
+
+#if 1
+#define GetCoreDataFromValue(obj, type, ptr) do { \
+ ptr = (type*)DATA_PTR(obj); \
+} while (0)
+#else
+#define GetCoreDataFromValue(obj, type, ptr) Data_Get_Struct(obj, type, ptr)
+#endif
+
+#define GetISeqPtr(obj, ptr) \
+ GetCoreDataFromValue(obj, mrb_iseq_t, ptr)
+
+struct mrb_iseq_struct;
+
+//enum ruby_special_exceptions {
+// ruby_error_reenter,
+// ruby_error_nomemory,
+// ruby_error_sysstack,
+// ruby_special_error_count
+//};
+
+#define GetVMPtr(obj, ptr) \
+ GetCoreDataFromValue(obj, mrb_vm_t, ptr)
+
+#if defined(ENABLE_VM_OBJSPACE) && ENABLE_VM_OBJSPACE
+struct mrb_objspace;
+void mrb_objspace_free(struct mrb_objspace *);
+#endif
+
+typedef struct mrb_block_struct {
+ mrb_value self; /* share with method frame if it's only block */
+ mrb_value *lfp; /* share with method frame if it's only block */
+ mrb_value *dfp; /* share with method frame if it's only block */
+ mrb_iseq_t *iseq;
+ mrb_value proc;
+} mrb_block_t;
+
+#define GetThreadPtr(obj, ptr) \
+ GetCoreDataFromValue(obj, mrb_thread_t, ptr)
+
+//typedef RUBY_JMP_BUF mrb_jmpbuf_t; /* kusuda */
+#define mrb_jmpbuf_t void* /* kusuda */
+
+struct mrb_vm_protect_tag {
+ struct mrb_vm_protect_tag *prev;
+};
+
+#define RUBY_VM_VALUE_CACHE_SIZE 0x1000
+#define USE_VALUE_CACHE 0
+
+struct mrb_mutex_struct;
+
+
+/* iseq.c */
+mrb_value mrb_iseq_new(NODE*, mrb_value, mrb_value, mrb_value, mrb_value, mrb_value);
+mrb_value mrb_iseq_new_top(NODE *node, mrb_value name, mrb_value filename, mrb_value filepath, mrb_value parent);
+mrb_value mrb_iseq_new_main(NODE *node, mrb_value filename, mrb_value filepath);
+mrb_value mrb_iseq_new_with_bopt(NODE*, mrb_value, mrb_value, mrb_value, mrb_value, mrb_value, mrb_value, mrb_value);
+mrb_value mrb_iseq_new_with_opt(NODE*, mrb_value, mrb_value, mrb_value, mrb_value, mrb_value, mrb_value, const mrb_compile_option_t*);
+mrb_value mrb_iseq_compile(mrb_value src, mrb_value file, mrb_value line);
+mrb_value mrb_iseq_disasm(mrb_value self);
+int mrb_iseq_disasm_insn(mrb_value str, mrb_value *iseqval, size_t pos, mrb_iseq_t *iseq, mrb_value child);
+const char *ruby_node_name(int node);
+int mrb_iseq_first_lineno(mrb_iseq_t *iseq);
+
+RUBY_EXTERN mrb_value mrb_cISeq;
+RUBY_EXTERN mrb_value mrb_cRubyVM;
+RUBY_EXTERN mrb_value mrb_cEnv;
+RUBY_EXTERN mrb_value mrb_mRubyVMFrozenCore;
+
+/* each thread has this size stack : 128KB */
+#define RUBY_VM_THREAD_STACK_SIZE (128 * 1024)
+
+#define GetProcPtr(obj, ptr) \
+ GetCoreDataFromValue(obj, mrb_proc_t, ptr)
+
+typedef struct {
+ mrb_block_t block;
+
+ mrb_value envval; /* for GC mark */
+ mrb_value blockprocval;
+ int safe_level;
+ int is_from_method;
+ int is_lambda;
+} mrb_proc_t;
+
+#define GetEnvPtr(obj, ptr) \
+ GetCoreDataFromValue(obj, mrb_env_t, ptr)
+
+typedef struct {
+ mrb_value *env;
+ int env_size;
+ int local_size;
+ mrb_value prev_envval; /* for GC mark */
+ mrb_block_t block;
+} mrb_env_t;
+
+//#define GetBindingPtr(obj, ptr)
+// GetCoreDataFromValue(obj, mrb_binding_t, ptr)
+
+//typedef struct {
+// mrb_value env;
+// mrb_value filename;
+// unsigned short line_no;
+//} mrb_binding_t;
+
+/* used by compile time and send insn */
+#define VM_CALL_ARGS_SPLAT_BIT (0x01 << 1)
+#define VM_CALL_ARGS_BLOCKARG_BIT (0x01 << 2)
+#define VM_CALL_FCALL_BIT (0x01 << 3)
+#define VM_CALL_VCALL_BIT (0x01 << 4)
+#define VM_CALL_TAILCALL_BIT (0x01 << 5)
+#define VM_CALL_TAILRECURSION_BIT (0x01 << 6)
+#define VM_CALL_SUPER_BIT (0x01 << 7)
+#define VM_CALL_OPT_SEND_BIT (0x01 << 8)
+
+#define VM_SPECIAL_OBJECT_VMCORE 0x01
+#define VM_SPECIAL_OBJECT_CBASE 0x02
+#define VM_SPECIAL_OBJECT_CONST_BASE 0x03
+
+#define VM_FRAME_MAGIC_METHOD 0x11
+#define VM_FRAME_MAGIC_BLOCK 0x21
+#define VM_FRAME_MAGIC_CLASS 0x31
+#define VM_FRAME_MAGIC_TOP 0x41
+#define VM_FRAME_MAGIC_FINISH 0x51
+#define VM_FRAME_MAGIC_CFUNC 0x61
+#define VM_FRAME_MAGIC_PROC 0x71
+#define VM_FRAME_MAGIC_IFUNC 0x81
+#define VM_FRAME_MAGIC_EVAL 0x91
+#define VM_FRAME_MAGIC_LAMBDA 0xa1
+#define VM_FRAME_MAGIC_MASK_BITS 8
+#define VM_FRAME_MAGIC_MASK (~(~0<<VM_FRAME_MAGIC_MASK_BITS))
+
+#define VM_FRAME_TYPE(cfp) ((cfp)->flag & VM_FRAME_MAGIC_MASK)
+
+/* other frame flag */
+#define VM_FRAME_FLAG_PASSED 0x0100
+
+#define RUBYVM_CFUNC_FRAME_P(cfp) \
+ (VM_FRAME_TYPE(cfp) == VM_FRAME_MAGIC_CFUNC)
+
+/* inline cache */
+typedef struct iseq_inline_cache_entry *IC;
+
+extern mrb_value ruby_vm_global_state_version;
+
+#define GET_VM_STATE_VERSION() (ruby_vm_global_state_version)
+#define INC_VM_STATE_VERSION() \
+ (ruby_vm_global_state_version = (ruby_vm_global_state_version+1) & 0x8fffffff)
+void mrb_vm_change_state(void);
+
+typedef mrb_value CDHASH;
+
+#define GC_GUARDED_PTR(p) ((mrb_value)((mrb_value)(p) | 0x01))
+#define GC_GUARDED_PTR_REF(p) ((void *)(((mrb_value)p) & ~0x03))
+#define GC_GUARDED_PTR_P(p) (((mrb_value)p) & 0x01)
+
+#define RUBY_VM_PREVIOUS_CONTROL_FRAME(cfp) (cfp+1)
+#define RUBY_VM_NEXT_CONTROL_FRAME(cfp) (cfp-1)
+#define RUBY_VM_END_CONTROL_FRAME(th) \
+ ((mrb_control_frame_t *)((th)->stack + (th)->stack_size))
+#define RUBY_VM_VALID_CONTROL_FRAME_P(cfp, ecfp) \
+ ((void *)(ecfp) > (void *)(cfp))
+#define RUBY_VM_CONTROL_FRAME_STACK_OVERFLOW_P(th, cfp) \
+ (!RUBY_VM_VALID_CONTROL_FRAME_P((cfp), RUBY_VM_END_CONTROL_FRAME(th)))
+
+#define RUBY_VM_IFUNC_P(ptr) (BUILTIN_TYPE(ptr) == T_NODE)
+#define RUBY_VM_NORMAL_ISEQ_P(ptr) \
+ (ptr && !RUBY_VM_IFUNC_P(ptr))
+
+#define RUBY_VM_GET_BLOCK_PTR_IN_CFP(cfp) ((mrb_block_t *)(&(cfp)->self))
+#define RUBY_VM_GET_CFP_FROM_BLOCK_PTR(b) \
+ ((mrb_control_frame_t *)((mrb_value *)(b) - 5))
+
+/* VM related object allocate functions */
+//mrb_value mrb_thread_alloc(mrb_value klass);
+mrb_value mrb_proc_alloc(mrb_value klass);
+
+/* for debug */
+extern void mrb_vmdebug_stack_dump_raw(mrb_thread_t *, mrb_control_frame_t *);
+#define SDR() mrb_vmdebug_stack_dump_raw(GET_THREAD(), GET_THREAD()->cfp)
+#define SDR2(cfp) mrb_vmdebug_stack_dump_raw(GET_THREAD(), (cfp))
+void mrb_vm_bugreport(void);
+
+/* functions about thread/vm execution */
+mrb_value mrb_iseq_eval(mrb_value iseqval);
+mrb_value mrb_iseq_eval_main(mrb_value iseqval);
+void mrb_enable_interrupt(void);
+void mrb_disable_interrupt(void);
+//int mrb_thread_method_id_and_class(mrb_thread_t *th, mrb_sym *idp, mrb_value *klassp);
+
+mrb_value mrb_vm_invoke_proc(mrb_thread_t *th, mrb_proc_t *proc, mrb_value self,
+ int argc, const mrb_value *argv, const mrb_block_t *blockptr);
+mrb_value mrb_vm_make_proc(mrb_thread_t *th, const mrb_block_t *block, mrb_value klass);
+mrb_value mrb_vm_make_env_object(mrb_thread_t *th, mrb_control_frame_t *cfp);
+
+//void mrb_thread_start_timer_thread(void);
+//void mrb_thread_stop_timer_thread(void);
+//void mrb_thread_reset_timer_thread(void);
+//void *mrb_thread_call_with_gvl(void *(*func)(void *), void *data1);
+int ruby_thread_has_gvl_p(void);
+mrb_value mrb_make_backtrace(void);
+typedef int mrb_backtrace_iter_func(void *, mrb_value, int, mrb_value);
+int mrb_backtrace_each(mrb_backtrace_iter_func *iter, void *arg);
+//mrb_control_frame_t *mrb_vm_get_ruby_level_next_cfp(mrb_thread_t *th, mrb_control_frame_t *cfp);
+int mrb_vm_get_sourceline(const mrb_control_frame_t *);
+mrb_value mrb_name_err_mesg_new(mrb_value obj, mrb_value mesg, mrb_value recv, mrb_value method);
+
+NOINLINE(void mrb_gc_save_machine_context(mrb_thread_t *));
+
+//#define sysstack_error GET_VM()->special_exceptions[ruby_error_sysstack]
+
+mrb_value mrb_str_resurrect(mrb_value str);
+mrb_value mrb_ary_resurrect(mrb_value ary);
+
+/* for thread */
+
+#if RUBY_VM_THREAD_MODEL == 2
+RUBY_EXTERN mrb_thread_t *ruby_current_thread;
+extern mrb_vm_t *ruby_current_vm;
+
+#define GET_VM() ruby_current_vm
+#define GET_THREAD() ruby_current_thread
+#define mrb_thread_set_current_raw(th) (void)(ruby_current_thread = (th))
+#define mrb_thread_set_current(th) do { \
+ mrb_thread_set_current_raw(th); \
+ th->vm->running_thread = th; \
+} while (0)
+
+#else
+#error "unsupported thread model"
+#endif
+
+#define RUBY_VM_SET_INTERRUPT(th) ((th)->interrupt_flag |= 0x02)
+#define RUBY_VM_SET_TIMER_INTERRUPT(th) ((th)->interrupt_flag |= 0x01)
+#define RUBY_VM_SET_FINALIZER_INTERRUPT(th) ((th)->interrupt_flag |= 0x04)
+#define RUBY_VM_INTERRUPTED(th) ((th)->interrupt_flag & 0x02)
+
+void mrb_threadptr_check_signal(mrb_thread_t *mth);
+//void mrb_threadptr_signal_raise(mrb_thread_t *th, int sig);
+void mrb_threadptr_signal_exit(mrb_state *mrb, mrb_thread_t *th);
+//void mrb_threadptr_execute_interrupts(mrb_thread_t *);
+
+void mrb_thread_lock_unlock(mrb_thread_lock_t *);
+void mrb_thread_lock_destroy(mrb_thread_lock_t *);
+
+//#define RUBY_VM_CHECK_INTS_TH(th) do { \
+// if (UNLIKELY(th->interrupt_flag)) { \
+// mrb_threadptr_execute_interrupts(th); \
+// } \
+//} while (0)
+
+//#define RUBY_VM_CHECK_INTS() \
+// RUBY_VM_CHECK_INTS_TH(GET_THREAD())
+
+/* tracer */
+//void
+//mrb_threadptr_exec_event_hooks(mrb_thread_t *th, mrb_event_flag_t flag, mrb_value self, mrb_sym id, mrb_value klass);
+#if 0
+#define EXEC_EVENT_HOOK(th, flag, self, id, klass) do { \
+ mrb_event_flag_t wait_event__ = th->event_flags; \
+ if (UNLIKELY(wait_event__)) { \
+ if (wait_event__ & (flag | RUBY_EVENT_VM)) { \
+ mrb_threadptr_exec_event_hooks(th, flag, self, id, klass); \
+ } \
+ } \
+} while (0)
+#endif
+#endif /* RUBY_VM_CORE_H */