2007-06-24 20:45:48 -04:00
|
|
|
/* -*-c-*- */
|
2006-12-31 10:02:22 -05:00
|
|
|
/*
|
|
|
|
* from eval.c
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "eval_intern.h"
|
|
|
|
|
|
|
|
NORETURN(static VALUE rb_f_throw _((int, VALUE *)));
|
|
|
|
|
|
|
|
/*
|
|
|
|
* call-seq:
|
|
|
|
* throw(symbol [, obj])
|
* compile.c, dir.c, eval.c, eval_jump.h, eval_method.h, numeric.c,
pack.c, parse.y, re.c, thread.c, vm.c, vm_dump.c, call_cfunc.ci,
thread_pthread.ci, thread_win32.ci: fixed indentation.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@12431 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2007-06-05 00:25:10 -04:00
|
|
|
*
|
2006-12-31 10:02:22 -05:00
|
|
|
* Transfers control to the end of the active +catch+ block
|
|
|
|
* waiting for _symbol_. Raises +NameError+ if there
|
|
|
|
* is no +catch+ block for the symbol. The optional second
|
|
|
|
* parameter supplies a return value for the +catch+ block,
|
|
|
|
* which otherwise defaults to +nil+. For examples, see
|
|
|
|
* <code>Kernel::catch</code>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
static VALUE
|
|
|
|
rb_f_throw(int argc, VALUE *argv)
|
|
|
|
{
|
|
|
|
VALUE tag, value;
|
* blockinlining.c, error.c, eval.c, eval_error.h, eval_intern.h,
eval_jump.h, eval_load.c, eval_safe.h, gc.c, proc.c, signal.c,
thread.c, thread_pthread.ci, thread_win32.ci, vm.c, vm.h,
vm_dump.c, vm_evalbody.ci, yarvcore.c, yarvcore.h:
fix typo (rb_thead_t -> rb_thread_t).
* eval_intern.h: remove unused definitions.
* common.mk: fix around vm_opts.h path
and remove harmful argument passed to insns2vm.rb.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@11658 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2007-02-08 01:37:46 -05:00
|
|
|
rb_thread_t *th = GET_THREAD();
|
* blockinlining.c, compile.c, compile.h, error.c, eval.c,
eval_intern.h, eval_jump.h, eval_load.c, eval_method.h,
eval_safe.h, gc.c, insnhelper.h, insns.def, iseq.c, proc.c,
process.c, signal.c, thread.c, thread_pthread.ci, thread_win32.ci,
vm.c, vm.h, vm_dump.c, vm_evalbody.ci, vm_macro.def,
yarv.h, yarvcore.h, yarvcore.c: change type and macro names:
* yarv_*_t -> rb_*_t
* yarv_*_struct -> rb_*_struct
* yarv_tag -> rb_vm_tag
* YARV_* -> RUBY_VM_*
* proc.c, vm.c: move functions about env object creation
from proc.c to vm.c.
* proc.c, yarvcore.c: fix rb_cVM initialization place.
* inits.c: change Init_ISeq() order (after Init_VM).
* ruby.h, proc.c: change declaration place of rb_cEnv
from proc.c to ruby.c.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@11651 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2007-02-06 14:00:03 -05:00
|
|
|
struct rb_vm_tag *tt = th->tag;
|
2006-12-31 10:02:22 -05:00
|
|
|
|
|
|
|
rb_scan_args(argc, argv, "11", &tag, &value);
|
|
|
|
while (tt) {
|
|
|
|
if (tt->tag == tag) {
|
|
|
|
tt->retval = value;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
tt = tt->prev;
|
|
|
|
}
|
|
|
|
if (!tt) {
|
2007-09-01 08:56:29 -04:00
|
|
|
VALUE desc = rb_inspect(tag);
|
|
|
|
rb_raise(rb_eArgError, "uncaught throw %s", RSTRING_PTR(desc));
|
2006-12-31 10:02:22 -05:00
|
|
|
}
|
|
|
|
rb_trap_restore_mask();
|
2007-09-14 03:17:17 -04:00
|
|
|
th->errinfo = NEW_THROW_OBJECT(tag, 0, TAG_THROW);
|
2006-12-31 10:02:22 -05:00
|
|
|
|
|
|
|
JUMP_TAG(TAG_THROW);
|
|
|
|
#ifndef __GNUC__
|
|
|
|
return Qnil; /* not reached */
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
rb_throw(const char *tag, VALUE val)
|
|
|
|
{
|
|
|
|
VALUE argv[2];
|
|
|
|
|
|
|
|
argv[0] = ID2SYM(rb_intern(tag));
|
|
|
|
argv[1] = val;
|
|
|
|
rb_f_throw(2, argv);
|
|
|
|
}
|
|
|
|
|
2007-09-01 08:56:29 -04:00
|
|
|
void
|
|
|
|
rb_throw_obj(VALUE tag, VALUE val)
|
|
|
|
{
|
|
|
|
VALUE argv[2];
|
|
|
|
|
|
|
|
argv[0] = tag;
|
|
|
|
argv[1] = val;
|
|
|
|
rb_f_throw(2, argv);
|
|
|
|
}
|
|
|
|
|
2006-12-31 10:02:22 -05:00
|
|
|
/*
|
|
|
|
* call-seq:
|
|
|
|
* catch(symbol) {| | block } > obj
|
* compile.c, dir.c, eval.c, eval_jump.h, eval_method.h, numeric.c,
pack.c, parse.y, re.c, thread.c, vm.c, vm_dump.c, call_cfunc.ci,
thread_pthread.ci, thread_win32.ci: fixed indentation.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@12431 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2007-06-05 00:25:10 -04:00
|
|
|
*
|
2006-12-31 10:02:22 -05:00
|
|
|
* +catch+ executes its block. If a +throw+ is
|
|
|
|
* executed, Ruby searches up its stack for a +catch+ block
|
|
|
|
* with a tag corresponding to the +throw+'s
|
|
|
|
* _symbol_. If found, that block is terminated, and
|
|
|
|
* +catch+ returns the value given to +throw+. If
|
|
|
|
* +throw+ is not called, the block terminates normally, and
|
|
|
|
* the value of +catch+ is the value of the last expression
|
|
|
|
* evaluated. +catch+ expressions may be nested, and the
|
|
|
|
* +throw+ call need not be in lexical scope.
|
* compile.c, dir.c, eval.c, eval_jump.h, eval_method.h, numeric.c,
pack.c, parse.y, re.c, thread.c, vm.c, vm_dump.c, call_cfunc.ci,
thread_pthread.ci, thread_win32.ci: fixed indentation.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@12431 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2007-06-05 00:25:10 -04:00
|
|
|
*
|
2006-12-31 10:02:22 -05:00
|
|
|
* def routine(n)
|
|
|
|
* puts n
|
|
|
|
* throw :done if n <= 0
|
|
|
|
* routine(n-1)
|
|
|
|
* end
|
* compile.c, dir.c, eval.c, eval_jump.h, eval_method.h, numeric.c,
pack.c, parse.y, re.c, thread.c, vm.c, vm_dump.c, call_cfunc.ci,
thread_pthread.ci, thread_win32.ci: fixed indentation.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@12431 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2007-06-05 00:25:10 -04:00
|
|
|
*
|
|
|
|
*
|
2006-12-31 10:02:22 -05:00
|
|
|
* catch(:done) { routine(3) }
|
* compile.c, dir.c, eval.c, eval_jump.h, eval_method.h, numeric.c,
pack.c, parse.y, re.c, thread.c, vm.c, vm_dump.c, call_cfunc.ci,
thread_pthread.ci, thread_win32.ci: fixed indentation.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@12431 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2007-06-05 00:25:10 -04:00
|
|
|
*
|
2006-12-31 10:02:22 -05:00
|
|
|
* <em>produces:</em>
|
* compile.c, dir.c, eval.c, eval_jump.h, eval_method.h, numeric.c,
pack.c, parse.y, re.c, thread.c, vm.c, vm_dump.c, call_cfunc.ci,
thread_pthread.ci, thread_win32.ci: fixed indentation.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@12431 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2007-06-05 00:25:10 -04:00
|
|
|
*
|
2006-12-31 10:02:22 -05:00
|
|
|
* 3
|
|
|
|
* 2
|
|
|
|
* 1
|
|
|
|
* 0
|
|
|
|
*/
|
|
|
|
|
|
|
|
static VALUE
|
2007-09-01 08:56:29 -04:00
|
|
|
rb_f_catch(int argc, VALUE *argv)
|
2006-12-31 10:02:22 -05:00
|
|
|
{
|
2007-09-01 08:56:29 -04:00
|
|
|
VALUE tag;
|
2006-12-31 10:02:22 -05:00
|
|
|
int state;
|
|
|
|
VALUE val = Qnil; /* OK */
|
* blockinlining.c, error.c, eval.c, eval_error.h, eval_intern.h,
eval_jump.h, eval_load.c, eval_safe.h, gc.c, proc.c, signal.c,
thread.c, thread_pthread.ci, thread_win32.ci, vm.c, vm.h,
vm_dump.c, vm_evalbody.ci, yarvcore.c, yarvcore.h:
fix typo (rb_thead_t -> rb_thread_t).
* eval_intern.h: remove unused definitions.
* common.mk: fix around vm_opts.h path
and remove harmful argument passed to insns2vm.rb.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@11658 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2007-02-08 01:37:46 -05:00
|
|
|
rb_thread_t *th = GET_THREAD();
|
2006-12-31 10:02:22 -05:00
|
|
|
|
2007-09-01 08:56:29 -04:00
|
|
|
rb_scan_args(argc, argv, "01", &tag);
|
|
|
|
if (argc == 0) {
|
|
|
|
tag = rb_obj_alloc(rb_cObject);
|
|
|
|
}
|
2007-04-24 21:35:13 -04:00
|
|
|
PUSH_TAG();
|
2006-12-31 10:02:22 -05:00
|
|
|
|
|
|
|
th->tag->tag = tag;
|
|
|
|
|
|
|
|
if ((state = EXEC_TAG()) == 0) {
|
2007-06-05 13:26:00 -04:00
|
|
|
val = rb_yield_0(1, &tag);
|
2006-12-31 10:02:22 -05:00
|
|
|
}
|
2007-09-01 10:46:00 -04:00
|
|
|
else if (state == TAG_THROW && RNODE(th->errinfo)->u1.value == tag) {
|
2006-12-31 10:02:22 -05:00
|
|
|
val = th->tag->retval;
|
2007-06-08 03:10:50 -04:00
|
|
|
th->errinfo = Qnil;
|
2006-12-31 10:02:22 -05:00
|
|
|
state = 0;
|
|
|
|
}
|
|
|
|
POP_TAG();
|
|
|
|
if (state)
|
|
|
|
JUMP_TAG(state);
|
|
|
|
|
|
|
|
return val;
|
|
|
|
}
|
|
|
|
|
2007-09-01 08:56:29 -04:00
|
|
|
static VALUE
|
|
|
|
catch_null_i(VALUE dmy)
|
|
|
|
{
|
|
|
|
return rb_funcall(Qnil, rb_intern("catch"), 0, 0);
|
|
|
|
}
|
|
|
|
|
2006-12-31 10:02:22 -05:00
|
|
|
static VALUE
|
|
|
|
catch_i(VALUE tag)
|
|
|
|
{
|
|
|
|
return rb_funcall(Qnil, rb_intern("catch"), 1, tag);
|
|
|
|
}
|
|
|
|
|
|
|
|
VALUE
|
|
|
|
rb_catch(const char *tag, VALUE (*func)(), VALUE data)
|
|
|
|
{
|
2007-09-01 08:56:29 -04:00
|
|
|
if (!tag) {
|
|
|
|
return rb_iterate(catch_null_i, 0, func, data);
|
|
|
|
}
|
|
|
|
return rb_iterate(catch_i, ID2SYM(rb_intern(tag)), func, data);
|
|
|
|
}
|
|
|
|
|
|
|
|
VALUE
|
|
|
|
rb_catch_obj(VALUE tag, VALUE (*func)(), VALUE data)
|
|
|
|
{
|
|
|
|
return rb_iterate((VALUE (*)_((VALUE)))catch_i, tag, func, data);
|
2006-12-31 10:02:22 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* exit */
|
|
|
|
|
2007-12-20 02:07:35 -05:00
|
|
|
void
|
|
|
|
rb_call_end_proc(VALUE data)
|
2006-12-31 10:02:22 -05:00
|
|
|
{
|
2007-12-20 02:07:35 -05:00
|
|
|
rb_proc_call(data, rb_ary_new());
|
2006-12-31 10:02:22 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
* call-seq:
|
|
|
|
* at_exit { block } -> proc
|
* compile.c, dir.c, eval.c, eval_jump.h, eval_method.h, numeric.c,
pack.c, parse.y, re.c, thread.c, vm.c, vm_dump.c, call_cfunc.ci,
thread_pthread.ci, thread_win32.ci: fixed indentation.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@12431 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2007-06-05 00:25:10 -04:00
|
|
|
*
|
2006-12-31 10:02:22 -05:00
|
|
|
* Converts _block_ to a +Proc+ object (and therefore
|
|
|
|
* binds it at the point of call) and registers it for execution when
|
|
|
|
* the program exits. If multiple handlers are registered, they are
|
|
|
|
* executed in reverse order of registration.
|
* compile.c, dir.c, eval.c, eval_jump.h, eval_method.h, numeric.c,
pack.c, parse.y, re.c, thread.c, vm.c, vm_dump.c, call_cfunc.ci,
thread_pthread.ci, thread_win32.ci: fixed indentation.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@12431 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2007-06-05 00:25:10 -04:00
|
|
|
*
|
2006-12-31 10:02:22 -05:00
|
|
|
* def do_at_exit(str1)
|
|
|
|
* at_exit { print str1 }
|
|
|
|
* end
|
|
|
|
* at_exit { puts "cruel world" }
|
|
|
|
* do_at_exit("goodbye ")
|
|
|
|
* exit
|
* compile.c, dir.c, eval.c, eval_jump.h, eval_method.h, numeric.c,
pack.c, parse.y, re.c, thread.c, vm.c, vm_dump.c, call_cfunc.ci,
thread_pthread.ci, thread_win32.ci: fixed indentation.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@12431 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2007-06-05 00:25:10 -04:00
|
|
|
*
|
2006-12-31 10:02:22 -05:00
|
|
|
* <em>produces:</em>
|
* compile.c, dir.c, eval.c, eval_jump.h, eval_method.h, numeric.c,
pack.c, parse.y, re.c, thread.c, vm.c, vm_dump.c, call_cfunc.ci,
thread_pthread.ci, thread_win32.ci: fixed indentation.
git-svn-id: svn+ssh://ci.ruby-lang.org/ruby/trunk@12431 b2dd03c8-39d4-4d8f-98ff-823fe69b080e
2007-06-05 00:25:10 -04:00
|
|
|
*
|
2006-12-31 10:02:22 -05:00
|
|
|
* goodbye cruel world
|
|
|
|
*/
|
|
|
|
|
|
|
|
static VALUE
|
|
|
|
rb_f_at_exit(void)
|
|
|
|
{
|
|
|
|
VALUE proc;
|
|
|
|
|
|
|
|
if (!rb_block_given_p()) {
|
|
|
|
rb_raise(rb_eArgError, "called without a block");
|
|
|
|
}
|
|
|
|
proc = rb_block_proc();
|
2007-12-20 02:07:35 -05:00
|
|
|
rb_set_end_proc(rb_call_end_proc, proc);
|
2006-12-31 10:02:22 -05:00
|
|
|
return proc;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct end_proc_data {
|
|
|
|
void (*func) ();
|
|
|
|
VALUE data;
|
|
|
|
int safe;
|
|
|
|
struct end_proc_data *next;
|
|
|
|
};
|
|
|
|
|
|
|
|
static struct end_proc_data *end_procs, *ephemeral_end_procs, *tmp_end_procs;
|
|
|
|
|
|
|
|
void
|
|
|
|
rb_set_end_proc(void (*func)(VALUE), VALUE data)
|
|
|
|
{
|
|
|
|
struct end_proc_data *link = ALLOC(struct end_proc_data);
|
|
|
|
struct end_proc_data **list;
|
2007-02-25 11:29:26 -05:00
|
|
|
rb_thread_t *th = GET_THREAD();
|
2006-12-31 10:02:22 -05:00
|
|
|
|
2007-02-25 11:29:26 -05:00
|
|
|
if (th->top_wrapper) {
|
2006-12-31 10:02:22 -05:00
|
|
|
list = &ephemeral_end_procs;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
list = &end_procs;
|
|
|
|
}
|
|
|
|
link->next = *list;
|
|
|
|
link->func = func;
|
|
|
|
link->data = data;
|
|
|
|
link->safe = rb_safe_level();
|
|
|
|
*list = link;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
rb_mark_end_proc(void)
|
|
|
|
{
|
|
|
|
struct end_proc_data *link;
|
|
|
|
|
|
|
|
link = end_procs;
|
|
|
|
while (link) {
|
|
|
|
rb_gc_mark(link->data);
|
|
|
|
link = link->next;
|
|
|
|
}
|
|
|
|
link = ephemeral_end_procs;
|
|
|
|
while (link) {
|
|
|
|
rb_gc_mark(link->data);
|
|
|
|
link = link->next;
|
|
|
|
}
|
|
|
|
link = tmp_end_procs;
|
|
|
|
while (link) {
|
|
|
|
rb_gc_mark(link->data);
|
|
|
|
link = link->next;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
rb_exec_end_proc(void)
|
|
|
|
{
|
|
|
|
struct end_proc_data *link, *tmp;
|
|
|
|
int status;
|
|
|
|
volatile int safe = rb_safe_level();
|
|
|
|
|
|
|
|
while (ephemeral_end_procs) {
|
|
|
|
tmp_end_procs = link = ephemeral_end_procs;
|
|
|
|
ephemeral_end_procs = 0;
|
|
|
|
while (link) {
|
2007-04-24 21:35:13 -04:00
|
|
|
PUSH_TAG();
|
2006-12-31 10:02:22 -05:00
|
|
|
if ((status = EXEC_TAG()) == 0) {
|
|
|
|
rb_set_safe_level_force(link->safe);
|
|
|
|
(*link->func) (link->data);
|
|
|
|
}
|
|
|
|
POP_TAG();
|
|
|
|
if (status) {
|
|
|
|
error_handle(status);
|
|
|
|
}
|
|
|
|
tmp = link;
|
|
|
|
tmp_end_procs = link = link->next;
|
|
|
|
free(tmp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
while (end_procs) {
|
|
|
|
tmp_end_procs = link = end_procs;
|
|
|
|
end_procs = 0;
|
|
|
|
while (link) {
|
2007-04-24 21:35:13 -04:00
|
|
|
PUSH_TAG();
|
2006-12-31 10:02:22 -05:00
|
|
|
if ((status = EXEC_TAG()) == 0) {
|
|
|
|
rb_set_safe_level_force(link->safe);
|
|
|
|
(*link->func) (link->data);
|
|
|
|
}
|
|
|
|
POP_TAG();
|
|
|
|
if (status) {
|
|
|
|
error_handle(status);
|
|
|
|
}
|
|
|
|
tmp = link;
|
|
|
|
tmp_end_procs = link = link->next;
|
|
|
|
free(tmp);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
rb_set_safe_level_force(safe);
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
2007-03-07 19:23:14 -05:00
|
|
|
Init_jump(void)
|
2006-12-31 10:02:22 -05:00
|
|
|
{
|
2007-09-01 08:56:29 -04:00
|
|
|
rb_define_global_function("catch", rb_f_catch, -1);
|
2006-12-31 10:02:22 -05:00
|
|
|
rb_define_global_function("throw", rb_f_throw, -1);
|
|
|
|
rb_define_global_function("at_exit", rb_f_at_exit, 0);
|
|
|
|
}
|