Skip to content

Commit 77cc808

Browse files
committedDec 11, 2023
Fix rb_integer_pack() for values between 2**62 and 2**63
* Fixes #3352 * Add specs for it.
1 parent b7cb7fe commit 77cc808

File tree

3 files changed

+22
-1
lines changed

3 files changed

+22
-1
lines changed
 

‎lib/truffle/truffle/cext.rb

+4
Original file line numberDiff line numberDiff line change
@@ -844,6 +844,10 @@ def rb_intern(str)
844844
Primitive.string_to_symbol(str, true)
845845
end
846846

847+
def rb_int_cmp(a, b)
848+
a <=> b
849+
end
850+
847851
def rb_int_positive_pow(a, b)
848852
a ** b
849853
end

‎spec/ruby/optional/capi/integer_spec.rb

+17
Original file line numberDiff line numberDiff line change
@@ -140,6 +140,23 @@
140140
result.should == -1
141141
@words.should == "\x11\x32\x54\x76\x98\xBA\xDC\xFE"
142142
end
143+
144+
it "converts numbers near the fixnum limit successfully" do
145+
result = @s.rb_integer_pack(0x7123_4567_89ab_cdef, @words, 1, 8, 0,
146+
CApiIntegerSpecs::NATIVE|CApiIntegerSpecs::PACK_2COMP)
147+
result.should == 1
148+
@words.should == "\xEF\xCD\xAB\x89\x67\x45\x23\x71"
149+
150+
result = @s.rb_integer_pack(2**62-1, @words, 1, 8, 0,
151+
CApiIntegerSpecs::NATIVE|CApiIntegerSpecs::PACK_2COMP)
152+
result.should == 1
153+
@words.should == "\xFF\xFF\xFF\xFF\xFF\xFF\xFF\x3F"
154+
155+
result = @s.rb_integer_pack(2**63-1, @words, 1, 8, 0,
156+
CApiIntegerSpecs::NATIVE|CApiIntegerSpecs::PACK_2COMP)
157+
result.should == 1
158+
@words.should == "\xFF\xFF\xFF\xFF\xFF\xFF\xFF\x7F"
159+
end
143160
end
144161
end
145162
end

‎src/main/c/cext/integer.c

+1-1
Original file line numberDiff line numberDiff line change
@@ -101,7 +101,7 @@ int rb_integer_pack(VALUE value, void *words, size_t numwords, size_t wordsize,
101101
long l = NUM2LONG(value);
102102
sign = (l > 0) - (l < 0);
103103
} else {
104-
sign = polyglot_as_i32(polyglot_invoke(rb_tr_unwrap(value), "<=>", 0));
104+
sign = polyglot_as_i32(RUBY_CEXT_INVOKE_NO_WRAP("rb_int_cmp", value, INT2FIX(0)));
105105
}
106106
int bytes_needed = size / 8 + (size % 8 == 0 ? 0 : 1);
107107
int words_needed = bytes_needed / wordsize + (bytes_needed % wordsize == 0 ? 0 : 1);

0 commit comments

Comments
 (0)
Please sign in to comment.