ruby/test/fiddle/test_closure.rb
Sutou Kouhei a47f153d9d
Import JRuby implementation (#147)
Fix GH-104

lib/fiddle/jruby.rb is based on

https://github.com/jruby/jruby/blob/master/lib/ruby/stdlib/fiddle/jruby.rb
.

Here are changes for it:

* Move `Fiddle::TYPE_*` to `Fiddle::Types::*`
* Add `Fiddle::Types::VARIADIC`
* Add `Fiddle::Types::CONST_STRING`
* Add `Fiddle::Types::BOOL`
* Add `Fiddle::Types::INT8_T`
* Add `Fiddle::Types::UINT8_T`
* Add `Fiddle::Types::INT16_T`
* Add `Fiddle::Types::UINT16_T`
* Add `Fiddle::Types::INT32_T`
* Add `Fiddle::Types::UINT32_T`
* Add `Fiddle::Types::INT64_T`
* Add `Fiddle::Types::UINT64_T`
* Add more `Fiddle::ALIGN_*` for the above new `Fiddle::Types::*`
* Add more `Fiddle::SIZEOF_*` for the above new `Fiddle::Types::*`
* Add support for specifying type as not only `Fiddle::Types::*` but
also `Symbol` like `:int`
* Add support for variable size arguments in `Fiddle::Function`
* Add `Fiddle::Closure#free`
* Add `Fiddle::Closure#freed?`
* Add `Fiddle::Error` as base the error class
* Add `Fiddle::Pointer#call_free` and stop using `FFI::AutoPointer` in
`Fiddle::Pointer`
* Add support for `Fiddle::Pointer.malloc {}` `Fiddle::Pointer`
* Add support for `Fiddle::Pointer#free=`
* Add `Fiddle::Pointer#freed?`
* Use binary string not C string for `Fiddle::Pointer#[]`
* Add `Fiddle::Handle.sym_defined?`
* Add `Fiddle::Handle#sym_defined?`
* Add `Fiddle::Handle::DEFAULT`
* Add `Fiddle::ClearedReferenceError`
* Add no-op `Fiddle::Pinned`

Some features are still "not implemented". So there are some "omit"s for
JRuby in tests.
2024-10-10 10:05:52 +09:00

156 lines
3.9 KiB
Ruby

# frozen_string_literal: true
begin
require_relative 'helper'
rescue LoadError
end
module Fiddle
class TestClosure < Fiddle::TestCase
def teardown
super
# We can't use ObjectSpace with JRuby.
return if RUBY_ENGINE == "jruby"
# Ensure freeing all closures.
# See https://github.com/ruby/fiddle/issues/102#issuecomment-1241763091 .
not_freed_closures = []
ObjectSpace.each_object(Fiddle::Closure) do |closure|
not_freed_closures << closure unless closure.freed?
end
assert_equal([], not_freed_closures)
end
def test_argument_errors
assert_raise(TypeError) do
Closure.new(TYPE_INT, TYPE_INT)
end
assert_raise(TypeError) do
Closure.new('foo', [TYPE_INT])
end
assert_raise(TypeError) do
Closure.new(TYPE_INT, ['meow!'])
end
end
def test_call
closure_class = Class.new(Closure) do
def call
10
end
end
closure_class.create(TYPE_INT, []) do |closure|
func = Function.new(closure, [], TYPE_INT)
assert_equal 10, func.call
end
end
def test_returner
closure_class = Class.new(Closure) do
def call thing
thing
end
end
closure_class.create(TYPE_INT, [TYPE_INT]) do |closure|
func = Function.new(closure, [TYPE_INT], TYPE_INT)
assert_equal 10, func.call(10)
end
end
def test_const_string
if RUBY_ENGINE == "jruby"
omit("Closure with :const_string works but " +
"Function with :const_string doesn't work with JRuby")
end
closure_class = Class.new(Closure) do
def call(string)
@return_string = "Hello! #{string}"
@return_string
end
end
closure_class.create(:const_string, [:const_string]) do |closure|
func = Function.new(closure, [:const_string], :const_string)
assert_equal("Hello! World!", func.call("World!"))
end
end
def test_bool
closure_class = Class.new(Closure) do
def call(bool)
not bool
end
end
closure_class.create(:bool, [:bool]) do |closure|
func = Function.new(closure, [:bool], :bool)
assert_equal(false, func.call(true))
end
end
def test_free
closure_class = Class.new(Closure) do
def call
10
end
end
closure_class.create(:int, [:void]) do |closure|
assert(!closure.freed?)
closure.free
assert(closure.freed?)
closure.free
end
end
def test_block_caller
cb = Closure::BlockCaller.new(TYPE_INT, [TYPE_INT]) do |one|
one
end
begin
func = Function.new(cb, [TYPE_INT], TYPE_INT)
assert_equal 11, func.call(11)
ensure
cb.free
end
end
def test_memsize_ruby_dev_42480
if RUBY_ENGINE == "jruby"
omit("We can't use ObjectSpace with JRuby")
end
require 'objspace'
n = 10000
n.times do
Closure.create(:int, [:void]) do |closure|
ObjectSpace.memsize_of(closure)
end
end
end
%w[INT SHORT CHAR LONG LONG_LONG].each do |name|
type = Fiddle.const_get("TYPE_#{name}") rescue next
size = Fiddle.const_get("SIZEOF_#{name}")
[[type, size-1, name], [-type, size, "unsigned_"+name]].each do |t, s, n|
define_method("test_conversion_#{n.downcase}") do
arg = nil
closure_class = Class.new(Closure) do
define_method(:call) {|x| arg = x}
end
closure_class.create(t, [t]) do |closure|
v = ~(~0 << (8*s))
arg = nil
assert_equal(v, closure.call(v))
assert_equal(arg, v, n)
arg = nil
func = Function.new(closure, [t], t)
assert_equal(v, func.call(v))
assert_equal(arg, v, n)
end
end
end
end
end
end if defined?(Fiddle)