module GirFFI # The InPointer class handles conversion from ruby types to pointers for # arguments with direction :in. This is used for arguments that are # arrays, strings, or interfaces. class InPointer < FFI::Pointer def self.from_array type, ary return nil if ary.nil? case type when :utf8, :filename from_utf8_array ary when :interface_pointer from_interface_pointer_array ary when Symbol from_basic_type_array type, ary when FFI::Enum from_enum_array type, ary when Array from_interface_pointer_array ary else raise NotImplementedError, type end end def self.from type, val return nil if val.nil? case type when Array _, sub_t = *type # TODO: Take array type into account (zero-terminated or not) self.from_array sub_t, val when :utf8, :filename from_utf8 val when :gint32, :gint8 self.new val when :void ArgHelper.object_to_inptr val else raise NotImplementedError end end class << self private def from_utf8_array ary ptr_ary = ary.map {|str| self.from :utf8, str} ptr_ary << nil self.from_array :pointer, ptr_ary end def from_interface_pointer_array ary ptr_ary = ary.map {|ifc| ifc.to_ptr} ptr_ary << nil self.from_array :pointer, ptr_ary end def from_enum_array type, ary self.from_array :int32, ary.map {|sym| type.to_native sym, nil } end def from_utf8 str len = str.bytesize ptr = AllocationHelper.safe_malloc(len + 1).write_string(str).put_char(len, 0) self.new ptr end def from_basic_type_array type, ary ffi_type = TypeMap.map_basic_type type block = ArgHelper.allocate_array_of_type ffi_type, ary.length + 1 block.send "put_array_of_#{ffi_type}", 0, ary block.send("put_#{ffi_type}", ary.length * FFI.type_size(ffi_type), (ffi_type == :pointer ? nil : 0)) self.new block end end end end