зеркало из https://github.com/github/ruby.git
415 строки
16 KiB
Ruby
415 строки
16 KiB
Ruby
# frozen_string_literal: true
|
|
begin
|
|
require_relative 'helper'
|
|
require 'fiddle/cparser'
|
|
require 'fiddle/import'
|
|
rescue LoadError
|
|
end
|
|
|
|
module Fiddle
|
|
class TestCParser < TestCase
|
|
include CParser
|
|
|
|
def test_char_ctype
|
|
assert_equal(TYPE_CHAR, parse_ctype('char'))
|
|
assert_equal(TYPE_CHAR, parse_ctype('const char'))
|
|
assert_equal(TYPE_CHAR, parse_ctype('signed char'))
|
|
assert_equal(TYPE_CHAR, parse_ctype('const signed char'))
|
|
assert_equal(-TYPE_CHAR, parse_ctype('unsigned char'))
|
|
assert_equal(-TYPE_CHAR, parse_ctype('const unsigned char'))
|
|
end
|
|
|
|
def test_short_ctype
|
|
assert_equal(TYPE_SHORT, parse_ctype('short'))
|
|
assert_equal(TYPE_SHORT, parse_ctype('const short'))
|
|
assert_equal(TYPE_SHORT, parse_ctype('short int'))
|
|
assert_equal(TYPE_SHORT, parse_ctype('const short int'))
|
|
assert_equal(TYPE_SHORT, parse_ctype('int short'))
|
|
assert_equal(TYPE_SHORT, parse_ctype('const int short'))
|
|
assert_equal(TYPE_SHORT, parse_ctype('signed short'))
|
|
assert_equal(TYPE_SHORT, parse_ctype('const signed short'))
|
|
assert_equal(TYPE_SHORT, parse_ctype('short signed'))
|
|
assert_equal(TYPE_SHORT, parse_ctype('const short signed'))
|
|
assert_equal(TYPE_SHORT, parse_ctype('signed short int'))
|
|
assert_equal(TYPE_SHORT, parse_ctype('const signed short int'))
|
|
assert_equal(TYPE_SHORT, parse_ctype('signed int short'))
|
|
assert_equal(TYPE_SHORT, parse_ctype('const signed int short'))
|
|
assert_equal(TYPE_SHORT, parse_ctype('int signed short'))
|
|
assert_equal(TYPE_SHORT, parse_ctype('const int signed short'))
|
|
assert_equal(TYPE_SHORT, parse_ctype('int short signed'))
|
|
assert_equal(TYPE_SHORT, parse_ctype('const int short signed'))
|
|
assert_equal(-TYPE_SHORT, parse_ctype('unsigned short'))
|
|
assert_equal(-TYPE_SHORT, parse_ctype('const unsigned short'))
|
|
assert_equal(-TYPE_SHORT, parse_ctype('unsigned short int'))
|
|
assert_equal(-TYPE_SHORT, parse_ctype('const unsigned short int'))
|
|
assert_equal(-TYPE_SHORT, parse_ctype('unsigned int short'))
|
|
assert_equal(-TYPE_SHORT, parse_ctype('const unsigned int short'))
|
|
assert_equal(-TYPE_SHORT, parse_ctype('short int unsigned'))
|
|
assert_equal(-TYPE_SHORT, parse_ctype('const short int unsigned'))
|
|
assert_equal(-TYPE_SHORT, parse_ctype('int unsigned short'))
|
|
assert_equal(-TYPE_SHORT, parse_ctype('const int unsigned short'))
|
|
assert_equal(-TYPE_SHORT, parse_ctype('int short unsigned'))
|
|
assert_equal(-TYPE_SHORT, parse_ctype('const int short unsigned'))
|
|
end
|
|
|
|
def test_int_ctype
|
|
assert_equal(TYPE_INT, parse_ctype('int'))
|
|
assert_equal(TYPE_INT, parse_ctype('const int'))
|
|
assert_equal(TYPE_INT, parse_ctype('signed int'))
|
|
assert_equal(TYPE_INT, parse_ctype('const signed int'))
|
|
assert_equal(-TYPE_INT, parse_ctype('uint'))
|
|
assert_equal(-TYPE_INT, parse_ctype('const uint'))
|
|
assert_equal(-TYPE_INT, parse_ctype('unsigned int'))
|
|
assert_equal(-TYPE_INT, parse_ctype('const unsigned int'))
|
|
end
|
|
|
|
def test_long_ctype
|
|
assert_equal(TYPE_LONG, parse_ctype('long'))
|
|
assert_equal(TYPE_LONG, parse_ctype('const long'))
|
|
assert_equal(TYPE_LONG, parse_ctype('long int'))
|
|
assert_equal(TYPE_LONG, parse_ctype('const long int'))
|
|
assert_equal(TYPE_LONG, parse_ctype('int long'))
|
|
assert_equal(TYPE_LONG, parse_ctype('const int long'))
|
|
assert_equal(TYPE_LONG, parse_ctype('signed long'))
|
|
assert_equal(TYPE_LONG, parse_ctype('const signed long'))
|
|
assert_equal(TYPE_LONG, parse_ctype('signed long int'))
|
|
assert_equal(TYPE_LONG, parse_ctype('const signed long int'))
|
|
assert_equal(TYPE_LONG, parse_ctype('signed int long'))
|
|
assert_equal(TYPE_LONG, parse_ctype('const signed int long'))
|
|
assert_equal(TYPE_LONG, parse_ctype('long signed'))
|
|
assert_equal(TYPE_LONG, parse_ctype('const long signed'))
|
|
assert_equal(TYPE_LONG, parse_ctype('long int signed'))
|
|
assert_equal(TYPE_LONG, parse_ctype('const long int signed'))
|
|
assert_equal(TYPE_LONG, parse_ctype('int long signed'))
|
|
assert_equal(TYPE_LONG, parse_ctype('const int long signed'))
|
|
assert_equal(-TYPE_LONG, parse_ctype('unsigned long'))
|
|
assert_equal(-TYPE_LONG, parse_ctype('const unsigned long'))
|
|
assert_equal(-TYPE_LONG, parse_ctype('unsigned long int'))
|
|
assert_equal(-TYPE_LONG, parse_ctype('const unsigned long int'))
|
|
assert_equal(-TYPE_LONG, parse_ctype('long int unsigned'))
|
|
assert_equal(-TYPE_LONG, parse_ctype('const long int unsigned'))
|
|
assert_equal(-TYPE_LONG, parse_ctype('unsigned int long'))
|
|
assert_equal(-TYPE_LONG, parse_ctype('const unsigned int long'))
|
|
assert_equal(-TYPE_LONG, parse_ctype('int unsigned long'))
|
|
assert_equal(-TYPE_LONG, parse_ctype('const int unsigned long'))
|
|
assert_equal(-TYPE_LONG, parse_ctype('int long unsigned'))
|
|
assert_equal(-TYPE_LONG, parse_ctype('const int long unsigned'))
|
|
end
|
|
|
|
def test_size_t_ctype
|
|
assert_equal(TYPE_SIZE_T, parse_ctype("size_t"))
|
|
assert_equal(TYPE_SIZE_T, parse_ctype("const size_t"))
|
|
end
|
|
|
|
def test_ssize_t_ctype
|
|
assert_equal(TYPE_SSIZE_T, parse_ctype("ssize_t"))
|
|
assert_equal(TYPE_SSIZE_T, parse_ctype("const ssize_t"))
|
|
end
|
|
|
|
def test_ptrdiff_t_ctype
|
|
assert_equal(TYPE_PTRDIFF_T, parse_ctype("ptrdiff_t"))
|
|
assert_equal(TYPE_PTRDIFF_T, parse_ctype("const ptrdiff_t"))
|
|
end
|
|
|
|
def test_intptr_t_ctype
|
|
assert_equal(TYPE_INTPTR_T, parse_ctype("intptr_t"))
|
|
assert_equal(TYPE_INTPTR_T, parse_ctype("const intptr_t"))
|
|
end
|
|
|
|
def test_uintptr_t_ctype
|
|
assert_equal(TYPE_UINTPTR_T, parse_ctype("uintptr_t"))
|
|
assert_equal(TYPE_UINTPTR_T, parse_ctype("const uintptr_t"))
|
|
end
|
|
|
|
def test_bool_ctype
|
|
assert_equal(TYPE_BOOL, parse_ctype('bool'))
|
|
end
|
|
|
|
def test_undefined_ctype
|
|
assert_raise(DLError) { parse_ctype('DWORD') }
|
|
end
|
|
|
|
def test_undefined_ctype_with_type_alias
|
|
assert_equal(-TYPE_LONG,
|
|
parse_ctype('DWORD', {"DWORD" => "unsigned long"}))
|
|
assert_equal(-TYPE_LONG,
|
|
parse_ctype('const DWORD', {"DWORD" => "unsigned long"}))
|
|
end
|
|
|
|
def expand_struct_types(types)
|
|
types.collect do |type|
|
|
case type
|
|
when Class
|
|
[expand_struct_types(type.types)]
|
|
when Array
|
|
[expand_struct_types([type[0]])[0][0], type[1]]
|
|
else
|
|
type
|
|
end
|
|
end
|
|
end
|
|
|
|
def test_struct_basic
|
|
assert_equal([[TYPE_INT, TYPE_CHAR], ['i', 'c']],
|
|
parse_struct_signature(['int i', 'char c']))
|
|
assert_equal([[TYPE_INT, TYPE_CHAR], ['i', 'c']],
|
|
parse_struct_signature(['const int i', 'const char c']))
|
|
end
|
|
|
|
def test_struct_array
|
|
assert_equal([[[TYPE_CHAR, 80], [TYPE_INT, 5]],
|
|
['buffer', 'x']],
|
|
parse_struct_signature(['char buffer[80]',
|
|
'int[5] x']))
|
|
assert_equal([[[TYPE_CHAR, 80], [TYPE_INT, 5]],
|
|
['buffer', 'x']],
|
|
parse_struct_signature(['const char buffer[80]',
|
|
'const int[5] x']))
|
|
end
|
|
|
|
def test_struct_nested_struct
|
|
types, members = parse_struct_signature([
|
|
'int x',
|
|
{inner: ['int i', 'char c']},
|
|
])
|
|
assert_equal([[TYPE_INT, [[TYPE_INT, TYPE_CHAR]]],
|
|
['x', ['inner', ['i', 'c']]]],
|
|
[expand_struct_types(types),
|
|
members])
|
|
end
|
|
|
|
def test_struct_nested_defined_struct
|
|
inner = Fiddle::Importer.struct(['int i', 'char c'])
|
|
assert_equal([[TYPE_INT, inner],
|
|
['x', ['inner', ['i', 'c']]]],
|
|
parse_struct_signature([
|
|
'int x',
|
|
{inner: inner},
|
|
]))
|
|
end
|
|
|
|
def test_struct_double_nested_struct
|
|
types, members = parse_struct_signature([
|
|
'int x',
|
|
{
|
|
outer: [
|
|
'int y',
|
|
{inner: ['int i', 'char c']},
|
|
],
|
|
},
|
|
])
|
|
assert_equal([[TYPE_INT, [[TYPE_INT, [[TYPE_INT, TYPE_CHAR]]]]],
|
|
['x', ['outer', ['y', ['inner', ['i', 'c']]]]]],
|
|
[expand_struct_types(types),
|
|
members])
|
|
end
|
|
|
|
def test_struct_nested_struct_array
|
|
types, members = parse_struct_signature([
|
|
'int x',
|
|
{
|
|
'inner[2]' => [
|
|
'int i',
|
|
'char c',
|
|
],
|
|
},
|
|
])
|
|
assert_equal([[TYPE_INT, [[TYPE_INT, TYPE_CHAR], 2]],
|
|
['x', ['inner', ['i', 'c']]]],
|
|
[expand_struct_types(types),
|
|
members])
|
|
end
|
|
|
|
def test_struct_double_nested_struct_inner_array
|
|
types, members = parse_struct_signature(outer: [
|
|
'int x',
|
|
{
|
|
'inner[2]' => [
|
|
'int i',
|
|
'char c',
|
|
],
|
|
},
|
|
])
|
|
assert_equal([[[[TYPE_INT, [[TYPE_INT, TYPE_CHAR], 2]]]],
|
|
[['outer', ['x', ['inner', ['i', 'c']]]]]],
|
|
[expand_struct_types(types),
|
|
members])
|
|
end
|
|
|
|
def test_struct_double_nested_struct_outer_array
|
|
types, members = parse_struct_signature([
|
|
'int x',
|
|
{
|
|
'outer[2]' => {
|
|
inner: [
|
|
'int i',
|
|
'char c',
|
|
],
|
|
},
|
|
},
|
|
])
|
|
assert_equal([[TYPE_INT, [[[[TYPE_INT, TYPE_CHAR]]], 2]],
|
|
['x', ['outer', [['inner', ['i', 'c']]]]]],
|
|
[expand_struct_types(types),
|
|
members])
|
|
end
|
|
|
|
def test_struct_array_str
|
|
assert_equal([[[TYPE_CHAR, 80], [TYPE_INT, 5]],
|
|
['buffer', 'x']],
|
|
parse_struct_signature('char buffer[80], int[5] x'))
|
|
assert_equal([[[TYPE_CHAR, 80], [TYPE_INT, 5]],
|
|
['buffer', 'x']],
|
|
parse_struct_signature('const char buffer[80], const int[5] x'))
|
|
end
|
|
|
|
def test_struct_function_pointer
|
|
assert_equal([[TYPE_VOIDP], ['cb']],
|
|
parse_struct_signature(['void (*cb)(const char*)']))
|
|
end
|
|
|
|
def test_struct_function_pointer_str
|
|
assert_equal([[TYPE_VOIDP, TYPE_VOIDP], ['cb', 'data']],
|
|
parse_struct_signature('void (*cb)(const char*), const char* data'))
|
|
end
|
|
|
|
def test_struct_string
|
|
assert_equal [[TYPE_INT,TYPE_VOIDP,TYPE_VOIDP], ['x', 'cb', 'name']], parse_struct_signature('int x; void (*cb)(); const char* name')
|
|
end
|
|
|
|
def test_struct_undefined
|
|
assert_raise(DLError) { parse_struct_signature(['int i', 'DWORD cb']) }
|
|
end
|
|
|
|
def test_struct_undefined_with_type_alias
|
|
assert_equal [[TYPE_INT,-TYPE_LONG], ['i', 'cb']], parse_struct_signature(['int i', 'DWORD cb'], {"DWORD" => "unsigned long"})
|
|
end
|
|
|
|
def test_signature_basic
|
|
func, ret, args = parse_signature('void func()')
|
|
assert_equal 'func', func
|
|
assert_equal TYPE_VOID, ret
|
|
assert_equal [], args
|
|
end
|
|
|
|
def test_signature_semi
|
|
func, ret, args = parse_signature('void func();')
|
|
assert_equal 'func', func
|
|
assert_equal TYPE_VOID, ret
|
|
assert_equal [], args
|
|
end
|
|
|
|
def test_signature_void_arg
|
|
func, ret, args = parse_signature('void func(void)')
|
|
assert_equal 'func', func
|
|
assert_equal TYPE_VOID, ret
|
|
assert_equal [], args
|
|
end
|
|
|
|
def test_signature_type_args
|
|
types = [
|
|
'char', 'unsigned char',
|
|
'short', 'unsigned short',
|
|
'int', 'unsigned int',
|
|
'long', 'unsigned long',
|
|
defined?(TYPE_LONG_LONG) && \
|
|
[
|
|
'long long', 'unsigned long long',
|
|
],
|
|
'float', 'double',
|
|
'const char*', 'void*',
|
|
].flatten.compact
|
|
func, ret, args = parse_signature("void func(#{types.join(',')})")
|
|
assert_equal 'func', func
|
|
assert_equal TYPE_VOID, ret
|
|
assert_equal [
|
|
TYPE_CHAR, -TYPE_CHAR,
|
|
TYPE_SHORT, -TYPE_SHORT,
|
|
TYPE_INT, -TYPE_INT,
|
|
TYPE_LONG, -TYPE_LONG,
|
|
defined?(TYPE_LONG_LONG) && \
|
|
[
|
|
TYPE_LONG_LONG, -TYPE_LONG_LONG,
|
|
],
|
|
TYPE_FLOAT, TYPE_DOUBLE,
|
|
TYPE_VOIDP, TYPE_VOIDP,
|
|
].flatten.compact, args
|
|
end
|
|
|
|
def test_signature_single_variable
|
|
func, ret, args = parse_signature('void func(int x)')
|
|
assert_equal 'func', func
|
|
assert_equal TYPE_VOID, ret
|
|
assert_equal [TYPE_INT], args
|
|
end
|
|
|
|
def test_signature_multiple_variables
|
|
func, ret, args = parse_signature('void func(int x, const char* s)')
|
|
assert_equal 'func', func
|
|
assert_equal TYPE_VOID, ret
|
|
assert_equal [TYPE_INT, TYPE_VOIDP], args
|
|
end
|
|
|
|
def test_signature_array_variable
|
|
func, ret, args = parse_signature('void func(int x[], int y[40])')
|
|
assert_equal 'func', func
|
|
assert_equal TYPE_VOID, ret
|
|
assert_equal [TYPE_VOIDP, TYPE_VOIDP], args
|
|
end
|
|
|
|
def test_signature_function_pointer
|
|
func, ret, args = parse_signature('int func(int (*sum)(int x, int y), int x, int y)')
|
|
assert_equal 'func', func
|
|
assert_equal TYPE_INT, ret
|
|
assert_equal [TYPE_VOIDP, TYPE_INT, TYPE_INT], args
|
|
end
|
|
|
|
def test_signature_variadic_arguments
|
|
unless Fiddle.const_defined?("TYPE_VARIADIC")
|
|
omit "libffi doesn't support variadic arguments"
|
|
end
|
|
assert_equal([
|
|
"printf",
|
|
TYPE_INT,
|
|
[TYPE_VOIDP, TYPE_VARIADIC],
|
|
],
|
|
parse_signature('int printf(const char *format, ...)'))
|
|
end
|
|
|
|
def test_signature_return_pointer
|
|
func, ret, args = parse_signature('void* malloc(size_t)')
|
|
assert_equal 'malloc', func
|
|
assert_equal TYPE_VOIDP, ret
|
|
assert_equal [TYPE_SIZE_T], args
|
|
end
|
|
|
|
def test_signature_return_array
|
|
func, ret, args = parse_signature('int (*func())[32]')
|
|
assert_equal 'func', func
|
|
assert_equal TYPE_VOIDP, ret
|
|
assert_equal [], args
|
|
end
|
|
|
|
def test_signature_return_array_with_args
|
|
func, ret, args = parse_signature('int (*func(const char* s))[]')
|
|
assert_equal 'func', func
|
|
assert_equal TYPE_VOIDP, ret
|
|
assert_equal [TYPE_VOIDP], args
|
|
end
|
|
|
|
def test_signature_return_function_pointer
|
|
func, ret, args = parse_signature('int (*func())(int x, int y)')
|
|
assert_equal 'func', func
|
|
assert_equal TYPE_VOIDP, ret
|
|
assert_equal [], args
|
|
end
|
|
|
|
def test_signature_return_function_pointer_with_args
|
|
func, ret, args = parse_signature('int (*func(int z))(int x, int y)')
|
|
assert_equal 'func', func
|
|
assert_equal TYPE_VOIDP, ret
|
|
assert_equal [TYPE_INT], args
|
|
end
|
|
end
|
|
end if defined?(Fiddle)
|