Module: GirFFI::InfoExt::ITypeInfo

Included in:
ErrorTypeInfo, ReceiverTypeInfo, UserDataTypeInfo
Defined in:
lib/gir_ffi/info_ext/i_type_info.rb

Overview

Extensions for GObjectIntrospection::ITypeInfo needed by GirFFI

Constant Summary collapse

TAG_TO_WRAPPER_CLASS_MAP =
{
  array:           'GLib::Array',
  byte_array:      'GLib::ByteArray',
  c:               'GirFFI::SizedArray',
  error:           'GLib::Error',
  ghash:           'GLib::HashTable',
  glist:           'GLib::List',
  gslist:          'GLib::SList',
  ptr_array:       'GLib::PtrArray',
  strv:            'GLib::Strv',
  utf8:            'GirFFI::InPointer',
  void:            'GirFFI::InPointer',
  zero_terminated: 'GirFFI::ZeroTerminated'
}.freeze
TAGS_NEEDING_RUBY_TO_C_CONVERSION =
[
  :array, :c, :callback, :error, :ghash, :glist, :gslist, :object,
  :ptr_array, :struct, :strv, :utf8, :void, :zero_terminated
].freeze
TAGS_NEEDING_C_TO_RUBY_CONVERSION =
[
  :array, :byte_array, :c, :error, :filename, :ghash, :glist, :gslist,
  :interface, :object, :ptr_array, :struct, :strv, :union, :utf8,
  :zero_terminated
].freeze
GOBJECT_VALUE_NAME =
'GObject::Value'.freeze

Class Method Summary collapse

Instance Method Summary collapse

Class Method Details

.flattened_tag_to_gtype_mapObject



8
9
10
11
12
13
14
15
16
17
18
19
20
21
# File 'lib/gir_ffi/info_ext/i_type_info.rb', line 8

def self.flattened_tag_to_gtype_map
  @flattened_tag_to_gtype_map ||= {
    array:    GObject::TYPE_ARRAY,
    c:        GObject::TYPE_POINTER,
    gboolean: GObject::TYPE_BOOLEAN,
    ghash:    GObject::TYPE_HASH_TABLE,
    gint32:   GObject::TYPE_INT,
    gint64:   GObject::TYPE_INT64,
    guint64:  GObject::TYPE_UINT64,
    strv:     GObject::TYPE_STRV,
    utf8:     GObject::TYPE_STRING,
    void:     GObject::TYPE_NONE
  }.freeze
end

Instance Method Details

#argument_class_nameObject

TODO: Use class rather than class name



90
91
92
93
94
95
96
# File 'lib/gir_ffi/info_ext/i_type_info.rb', line 90

def argument_class_name
  if tag == :interface
    interface.full_type_name
  else
    TAG_TO_WRAPPER_CLASS_MAP[flattened_tag]
  end
end

#element_typeObject



34
35
36
37
38
39
40
41
# File 'lib/gir_ffi/info_ext/i_type_info.rb', line 34

def element_type
  case tag
  when :glist, :gslist, :array, :c
    enumerable_element_type
  when :ghash
    dictionary_element_type
  end
end

#extra_conversion_argumentsObject



168
169
170
171
172
173
174
175
176
177
178
179
# File 'lib/gir_ffi/info_ext/i_type_info.rb', line 168

def extra_conversion_arguments
  case flattened_tag
  when :utf8, :void
    [flattened_tag]
  when :c
    [element_type, array_fixed_size]
  when :array, :ghash, :glist, :gslist, :ptr_array, :zero_terminated
    [element_type]
  else
    []
  end
end

#flattened_tagObject



43
44
45
46
47
48
49
50
51
52
# File 'lib/gir_ffi/info_ext/i_type_info.rb', line 43

def flattened_tag
  case tag
  when :interface
    interface_type
  when :array
    flattened_array_type
  else
    tag
  end
end

#gtypeObject



23
24
25
26
27
28
# File 'lib/gir_ffi/info_ext/i_type_info.rb', line 23

def gtype
  return interface.gtype if tag == :interface
  type = ITypeInfo.flattened_tag_to_gtype_map[flattened_tag]
  return type if type
  raise "Can't find GType for #{flattened_tag} pointer? = #{pointer?}"
end

#gvalue?Boolean

Returns:

  • (Boolean)


183
184
185
# File 'lib/gir_ffi/info_ext/i_type_info.rb', line 183

def gvalue?
  argument_class_name == GOBJECT_VALUE_NAME
end

#interface_typeObject



54
55
56
# File 'lib/gir_ffi/info_ext/i_type_info.rb', line 54

def interface_type
  tag == :interface && interface.info_type
end

#make_g_valueObject



30
31
32
# File 'lib/gir_ffi/info_ext/i_type_info.rb', line 30

def make_g_value
  GObject::Value.for_gtype gtype
end

#needs_c_to_ruby_conversion_for_callbacks?Boolean

Returns:

  • (Boolean)


155
156
157
158
# File 'lib/gir_ffi/info_ext/i_type_info.rb', line 155

def needs_c_to_ruby_conversion_for_callbacks?
  [:callback, :enum].include?(flattened_tag) ||
    needs_c_to_ruby_conversion_for_functions?
end

#needs_c_to_ruby_conversion_for_closures?Boolean

Returns:

  • (Boolean)


160
161
162
# File 'lib/gir_ffi/info_ext/i_type_info.rb', line 160

def needs_c_to_ruby_conversion_for_closures?
  [:array, :c, :ghash, :struct, :strv].include?(flattened_tag)
end

#needs_c_to_ruby_conversion_for_functions?Boolean

Returns:

  • (Boolean)


146
147
148
# File 'lib/gir_ffi/info_ext/i_type_info.rb', line 146

def needs_c_to_ruby_conversion_for_functions?
  TAGS_NEEDING_C_TO_RUBY_CONVERSION.include?(flattened_tag)
end

#needs_ruby_to_c_conversion_for_callbacks?Boolean

Returns:

  • (Boolean)


150
151
152
153
# File 'lib/gir_ffi/info_ext/i_type_info.rb', line 150

def needs_ruby_to_c_conversion_for_callbacks?
  [:enum].include?(flattened_tag) ||
    needs_ruby_to_c_conversion_for_functions?
end

#needs_ruby_to_c_conversion_for_closures?Boolean

Returns:

  • (Boolean)


164
165
166
# File 'lib/gir_ffi/info_ext/i_type_info.rb', line 164

def needs_ruby_to_c_conversion_for_closures?
  [:array].include?(flattened_tag)
end

#needs_ruby_to_c_conversion_for_functions?Boolean

Returns:

  • (Boolean)


142
143
144
# File 'lib/gir_ffi/info_ext/i_type_info.rb', line 142

def needs_ruby_to_c_conversion_for_functions?
  TAGS_NEEDING_RUBY_TO_C_CONVERSION.include?(flattened_tag)
end

#tag_or_classObject



58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
# File 'lib/gir_ffi/info_ext/i_type_info.rb', line 58

def tag_or_class
  base = case tag
         when:interface
           Builder.build_class interface
         when :ghash
           [tag, *element_type]
         else
           flattened_tag
         end
  if pointer? && tag != :utf8 && tag != :filename
    [:pointer, base]
  else
    base
  end
end

#to_callback_ffi_typeObject



111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
# File 'lib/gir_ffi/info_ext/i_type_info.rb', line 111

def to_callback_ffi_type
  return :pointer if pointer?

  case tag
  when :interface
    # TODO: Move this logic into interface
    case interface.info_type
    when :enum, :flags
      :int32
    else
      :pointer
    end
  when :gboolean
    # TODO: Move this logic into TypeMap
    :bool
  else
    TypeMap.map_basic_type tag
  end
end

#to_ffi_typeObject



98
99
100
101
102
103
104
105
106
107
108
109
# File 'lib/gir_ffi/info_ext/i_type_info.rb', line 98

def to_ffi_type
  return :pointer if pointer?

  case tag
  when :interface
    interface.to_ffi_type
  when :array
    [subtype_ffi_type(0), array_fixed_size]
  else
    TypeMap.map_basic_type tag
  end
end