1
0
Fork 0
mirror of https://github.com/ruby/ruby.git synced 2022-11-09 12:17:21 -05:00

[ruby/fiddle] Add "offsetof" to Struct classes (https://github.com/ruby/fiddle/pull/83)

* Add "offsetof" to Struct classes

I need to get the offset of a member inside a struct without allocating
the struct.  This patch adds an "offsetof" class method to structs that
are generated.

The usage is like this:

```ruby
MyStruct = struct [
  "int64_t i",
  "char c",
]

MyStruct.offsetof("i") # => 0
MyStruct.offsetof("c") # => 8
```

* Update test/fiddle/test_c_struct_builder.rb

Co-authored-by: Sutou Kouhei <kou@cozmixng.org>

https://github.com/ruby/fiddle/commit/4e3b60c5b6

Co-authored-by: Sutou Kouhei <kou@cozmixng.org>
This commit is contained in:
Aaron Patterson 2021-06-30 17:35:04 -07:00 committed by Nobuyoshi Nakada
parent a2c9e1b58a
commit 5c0d8c6369
No known key found for this signature in database
GPG key ID: 7CD2805BFA3770C6
2 changed files with 79 additions and 0 deletions

View file

@ -13,6 +13,30 @@ module Fiddle
CStructEntity
end
def self.offsetof(name, members, types) # :nodoc:
offset = 0
index = 0
member_index = members.index(name)
types.each { |type, count = 1|
orig_offset = offset
if type.respond_to?(:entity_class)
align = type.alignment
type_size = type.size
else
align = PackInfo::ALIGN_MAP[type]
type_size = PackInfo::SIZE_MAP[type]
end
offset = PackInfo.align(orig_offset, align)
return offset if index == member_index
offset += (type_size * count)
index += 1
}
nil
end
def each
return enum_for(__function__) unless block_given?
@ -75,6 +99,10 @@ module Fiddle
def CUnion.entity_class
CUnionEntity
end
def self.offsetof(name, members, types) # :nodoc:
0
end
end
# Wrapper for arrays within a struct
@ -172,6 +200,21 @@ module Fiddle
define_method(:to_i){ @entity.to_i }
define_singleton_method(:types) { types }
define_singleton_method(:members) { members }
# Return the offset of a struct member given its name.
# For example:
#
# MyStruct = struct [
# "int64_t i",
# "char c",
# ]
#
# MyStruct.offsetof("i") # => 0
# MyStruct.offsetof("c") # => 8
#
define_singleton_method(:offsetof) { |name|
klass.offsetof(name, members, types)
}
members.each{|name|
name = name[0] if name.is_a?(Array) # name is a nested struct
next if method_defined?(name)

View file

@ -0,0 +1,36 @@
# frozen_string_literal: true
begin
require_relative 'helper'
require 'fiddle/struct'
require 'fiddle/cparser'
rescue LoadError
end
module Fiddle
class TestCStructBuilder < TestCase
include Fiddle::CParser
def test_offsetof
types, members = parse_struct_signature(['int64_t i','char c'])
my_struct = Fiddle::CStructBuilder.create(Fiddle::CStruct, types, members)
assert_equal 0, my_struct.offsetof("i")
assert_equal Fiddle::SIZEOF_INT64_T, my_struct.offsetof("c")
end
def test_offset_with_gap
types, members = parse_struct_signature(['void *p', 'char c', 'long x'])
my_struct = Fiddle::CStructBuilder.create(Fiddle::CStruct, types, members)
assert_equal PackInfo.align(0, ALIGN_VOIDP), my_struct.offsetof("p")
assert_equal PackInfo.align(SIZEOF_VOIDP, ALIGN_CHAR), my_struct.offsetof("c")
assert_equal SIZEOF_VOIDP + PackInfo.align(SIZEOF_CHAR, ALIGN_LONG), my_struct.offsetof("x")
end
def test_union_offsetof
types, members = parse_struct_signature(['int64_t i','char c'])
my_struct = Fiddle::CStructBuilder.create(Fiddle::CUnion, types, members)
assert_equal 0, my_struct.offsetof("i")
assert_equal 0, my_struct.offsetof("c")
end
end
end if defined?(Fiddle)