forked from rubinius/rubinius
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathrange_mirror.rb
61 lines (51 loc) · 1.79 KB
/
range_mirror.rb
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
module Rubinius
class Mirror
class Range < Mirror
self.subject = ::Range
def excl
Rubinius.invoke_primitive :object_get_ivar, @object, :@excl
end
def step_float_iterations_size(first, last, step_size)
err = (first.abs + last.abs + (last - first).abs) / step_size.abs * Float::EPSILON
err = 0.5 if err > 0.5
if excl
iterations = ((last - first) / step_size - err).floor
iterations += 1 if iterations * step_size + first < last
else
iterations = ((last - first) / step_size + err).floor + 1
end
iterations
end
def step_iterations_size(first, last, step_size)
case first
when Float
step_float_iterations_size(first, last, step_size)
else
@object.size.nil? ? nil : (@object.size.fdiv(step_size)).ceil
end
end
def validate_step_size(first, last, step_size)
if step_size.kind_of? Float or first.kind_of? Float or last.kind_of? Float
# if any are floats they all must be
begin
step_size = Float(from = step_size)
first = Float(from = first)
last = Float(from = last)
rescue ArgumentError
raise TypeError, "no implicit conversion to float from #{from.class}"
end
else
step_size = Integer(from = step_size)
unless step_size.kind_of? Integer
raise TypeError, "can't convert #{from.class} to Integer (#{from.class}#to_int gives #{step_size.class})"
end
end
if step_size <= 0
raise ArgumentError, "step can't be negative" if step_size < 0
raise ArgumentError, "step can't be 0"
end
return first, last, step_size
end
end
end
end