summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--error.c6
-rw-r--r--test/ruby/test_exception.rb1
2 files changed, 6 insertions, 1 deletions
diff --git a/error.c b/error.c
index 2109936e72..7ec59cd9f4 100644
--- a/error.c
+++ b/error.c
@@ -342,7 +342,11 @@ rb_warn_m(int argc, VALUE *argv, VALUE exc)
uplevel = Qnil;
}
else if (!NIL_P(uplevel)) {
- uplevel = LONG2NUM((long)NUM2ULONG(uplevel) + 1);
+ long lev = NUM2LONG(uplevel);
+ if (lev < 0) {
+ rb_raise(rb_eArgError, "negative level (%ld)", lev);
+ }
+ uplevel = LONG2NUM(lev + 1);
uplevel = rb_vm_thread_backtrace_locations(1, &uplevel, GET_THREAD()->self);
if (!NIL_P(uplevel)) {
uplevel = rb_ary_entry(uplevel, 0);
diff --git a/test/ruby/test_exception.rb b/test/ruby/test_exception.rb
index 76be7f0ac2..212a52f24a 100644
--- a/test/ruby/test_exception.rb
+++ b/test/ruby/test_exception.rb
@@ -1007,6 +1007,7 @@ $stderr = $stdout; raise "\x82\xa0"') do |outs, errs, status|
def test_kernel_warn_uplevel
warning = capture_warning_warn {warn("test warning", uplevel: 0)}
assert_equal("#{__FILE__}:#{__LINE__-1}: warning: test warning\n", warning[0])
+ assert_raise(ArgumentError) {warn("test warning", uplevel: -1)}
end
def test_warning_warn_invalid_argument