ffi-bitfield 0.0.6 → 0.0.8

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 8e2586353fd6f53337bf3ffef8019d477c279c0284642165fad4d592315ccc1a
4
- data.tar.gz: 2772a6b9ef73faeccc105c501494d6fe82079614b6e23bab14de3954d1f1d1f2
3
+ metadata.gz: 50775cc156bce79d4d20153303484ba80d19d23707d5413ea2557676260e8b38
4
+ data.tar.gz: 941e057d7cd548da1fd133f81143e573d7795c060d7678334c74cd1ed7cdaa31
5
5
  SHA512:
6
- metadata.gz: 911015a3478c323f4469076489e68d7634d12188200dea1e8ae76263af368b992fb424c6f086cc46cb30aa501ca12326039a8a6a184524e2f54b6cb35f2495c1
7
- data.tar.gz: 82951de5f89c776408d35e656a8ce100c1306899e808569dd181cd5c56bfbee961329ac989455b9814e5815928227e732d4ea1e8d21a09141fadcb00cce49760
6
+ metadata.gz: 7329ad0a6bf66f580af0fdbbc3219774a53e7624a78e8e3891562150444282c0b2aae26b5eb4382edf8f67e27e66224aaf32592757f6e813105ab432a5e646a5
7
+ data.tar.gz: 3e4bb498d57114b6f9ccaf140d5789239609ff402f3be888d419fefa66957547659f1d5ab8577439f9fb98ad89fa6d56869f30a985b04486bc9bc90799c9b82a
data/README.md CHANGED
@@ -13,9 +13,21 @@ gem install ffi-bitfield
13
13
 
14
14
  ## Usage
15
15
 
16
+ Classes
17
+
18
+ * class BitStruct < FFI::Struct
19
+ * class ManagedBitStruct < FFI::ManagedStruct
20
+
21
+ Loading
22
+
16
23
  ```ruby
17
24
  require 'ffi/bit_struct'
25
+ require 'ffi/managed_bit_struct'
26
+ ```
27
+
28
+ Define your struct
18
29
 
30
+ ```ruby
19
31
  class Struct1 < FFI::BitStruct
20
32
  layout \
21
33
  :a, :uint8,
@@ -29,10 +41,14 @@ class Struct1 < FFI::BitStruct
29
41
  :y, 1,
30
42
  :z, 1
31
43
  end
44
+ ```
45
+
46
+ Reading
32
47
 
48
+ ```ruby
33
49
  s = Struct1.new
34
- s[:a] = 63
35
50
 
51
+ s[:a] = 63
36
52
  p s[:u] # 3
37
53
  p s[:v] # 3
38
54
  p s[:w] # 1
@@ -41,6 +57,19 @@ p s[:y] # 0
41
57
  p s[:z] # 0
42
58
  ```
43
59
 
60
+ Writing
61
+
62
+ ```ruby
63
+ s = Struct1.new
64
+
65
+ s[:u] = 0
66
+ s[:v] = 0
67
+ s[:w] = 0
68
+ s[:x] = 0
69
+ s[:y] = 1
70
+ p s[:a] # 64
71
+ ```
72
+
44
73
  ### Loading
45
74
 
46
75
  ```ruby
@@ -54,14 +83,6 @@ require 'ffi/bit_struct'
54
83
  require 'ffi/managed_bit_struct'
55
84
  ```
56
85
 
57
- ## API Overview
58
-
59
- ```md
60
- * module FFI
61
- * class BitStruct < FFI::Struct
62
- * class ManagedBitStruct < FFI::ManagedStruct
63
- ```
64
-
65
86
  ## Development
66
87
 
67
88
  ```
@@ -71,10 +92,18 @@ bundle install
71
92
  bundle exec rake test
72
93
  ```
73
94
 
95
+ * [ffi-bitfield - read/write bit fields with Ruby-FFI](https://dev.to/kojix2/ffi-bitfield-g4h)
96
+
74
97
  ## Contributing
75
98
 
99
+ Your feedback is important.
100
+
76
101
  ffi-bitfield is a library under development, so even small improvements like typofix are welcome! Please feel free to send us your pull requests.
77
- Bug reports and pull requests are welcome on GitHub at https://github.com/kojix2/bitstruct.
102
+ Bug reports and pull requests are welcome on GitHub at https://github.com/kojix2/ffi-bitfield.
103
+
104
+ Do you need commit rights to my repository?
105
+ Do you want to get admin rights and take over the project?
106
+ If so, please feel free to contact me @kojix2.
78
107
 
79
108
  ## License
80
109
 
@@ -2,10 +2,24 @@
2
2
 
3
3
  module FFI
4
4
  module BitField
5
- # Layout provides the `bit_fields` method for registering field members.
5
+ # Layout provides methods for defining bit field layouts.
6
+ # This module is extended by BitStruct and ManagedBitStruct classes.
6
7
  module Layout
7
- # @param [Array] layout_args
8
- # @return [Symbol] parent_name
8
+ # Defines bit fields within a parent field.
9
+ #
10
+ # @param [Array] layout_args An array where the first element is the parent field name,
11
+ # followed by alternating field name and bit width pairs
12
+ # @return [Symbol] parent_name The name of the parent field
13
+ #
14
+ # @example Define bit fields in an 8-bit integer
15
+ # bit_fields :flags,
16
+ # :read, 1, # 1 bit for read permission
17
+ # :write, 1, # 1 bit for write permission
18
+ # :execute, 1, # 1 bit for execute permission
19
+ # :unused, 5 # 5 unused bits
20
+ #
21
+ # @note The total bit width should not exceed the size of the parent field.
22
+ # For example, a :uint8 field can hold at most 8 bits.
9
23
  def bit_fields(*layout_args)
10
24
  # The reason for using class instance variable here instead of class variable
11
25
  # is not because class instance variables are clean,
@@ -2,36 +2,95 @@
2
2
 
3
3
  module FFI
4
4
  module BitField
5
- # Properties provides methods to read and write bit fields.
5
+ # Property provides methods for reading and writing bit field values.
6
+ # This module is included in BitStruct and ManagedBitStruct classes.
6
7
  module Property
7
- # @param [Symbol] member_name
8
- # @return [Integer] value
8
+ # Reads a value from a bit field or regular field.
9
+ #
10
+ # @param [Symbol] member_name The name of the field to read
11
+ # @return [Integer] The value of the field
12
+ #
13
+ # @example Reading a bit field
14
+ # struct[:flag1] # => 1
9
15
  def [](member_name)
10
16
  parent_name, start, width = member_value_info(member_name)
11
17
  if parent_name
12
18
  value = get_member_value(parent_name)
13
- value[start, width]
19
+ (value >> start) & ((1 << width) - 1)
14
20
  else
15
21
  get_member_value(member_name)
16
22
  end
17
23
  end
18
24
 
25
+ # Writes a value to a bit field or regular field.
26
+ #
27
+ # @param [Symbol] member_name The name of the field to write
28
+ # @param [Integer] value The value to write
29
+ # @return [Integer] The written value
30
+ # @raise [ArgumentError] If the value is too large for the bit field
31
+ # @raise [ArgumentError] If the value is too small (negative) for the bit field
32
+ # @raise [ArgumentError] If the member name is not a valid bit field
33
+ # @raise [TypeError] If the value is not an Integer
34
+ #
35
+ # @example Writing to a bit field
36
+ # struct[:flag1] = 1
37
+ # @example Writing a negative value (bit-flipped)
38
+ # struct[:field] = -1 # Sets all bits to 1
19
39
  def []=(member_name, value)
20
- parent_name, start, width = member_value_info(member_name)
21
- if parent_name
22
- raise "Value #{value} is larger than bit_length: #{width}" if value.bit_length > width
40
+ # Ensure value is an Integer
41
+ raise TypeError, "Value must be an Integer, got #{value.class}" unless value.is_a?(Integer)
23
42
 
24
- parent_value = get_member_value(parent_name)
25
- new_value = (((1 << width) - 1 << start) & parent_value ^ parent_value) |
26
- (value << start)
27
- set_member_value(parent_name, new_value)
28
- else
29
- set_member_value(member_name, value)
43
+ # Get bit field information
44
+ field_info = member_value_info(member_name)
45
+
46
+ # If not a bit field, delegate to regular field setter
47
+ return set_member_value(member_name, value) unless field_info
48
+
49
+ # Extract bit field information
50
+ parent_name, start, width = field_info
51
+
52
+ # Calculate max value for this bit width
53
+ max_value = (1 << width) - 1
54
+
55
+ # Handle negative values by bit-flipping
56
+ if value.negative?
57
+ # For negative values, we interpret them as bit-flipped positive values
58
+ # For example, with 4 bits, -1 becomes 1111 (15), -2 becomes 1110 (14), etc.
59
+
60
+ # Check if the negative value is within range
61
+ # For bit-flipping, valid range is -(2^n) to -1
62
+ min_value = -(1 << width)
63
+ if value < min_value
64
+ raise ArgumentError, "Value #{value} is too small for bit_length: #{width}, minimum is #{min_value}"
65
+ end
66
+
67
+ # Convert negative value to bit-flipped positive value
68
+ # -1 -> 15, -2 -> 14, etc.
69
+ value = max_value + value + 1
70
+
71
+ # Sanity check after conversion
72
+ if value.negative? || value > max_value
73
+ raise ArgumentError, "Internal error: converted value #{value} is out of range for bit_length: #{width}"
74
+ end
75
+ elsif value > max_value
76
+ # For positive values, check if they fit in the bit width
77
+ raise ArgumentError, "Value #{value} is too large for bit_length: #{width}, maximum is #{max_value}"
30
78
  end
79
+
80
+ # Update the parent field with the new bit field value
81
+ parent_value = get_member_value(parent_name)
82
+ mask = ((1 << width) - 1) << start
83
+ new_value = (parent_value & ~mask) | ((value & ((1 << width) - 1)) << start)
84
+
85
+ set_member_value(parent_name, new_value)
31
86
  end
32
87
 
33
88
  private
34
89
 
90
+ # Gets information about a bit field member.
91
+ #
92
+ # @param [Symbol] member_name The name of the bit field
93
+ # @return [Array, nil] An array containing [parent_name, start_bit, width] or nil if not a bit field
35
94
  def member_value_info(member_name)
36
95
  self.class.instance_variable_get(:@bit_field_hash_table)[member_name]
37
96
  end
@@ -2,6 +2,7 @@
2
2
 
3
3
  module FFI
4
4
  module BitField
5
- VERSION = '0.0.6'
5
+ # Current version of the ffi-bitfield gem.
6
+ VERSION = '0.0.8'
6
7
  end
7
8
  end
data/lib/ffi/bit_field.rb CHANGED
@@ -4,8 +4,26 @@ require_relative 'bit_field/version'
4
4
  require_relative 'bit_struct'
5
5
  require_relative 'managed_bit_struct'
6
6
 
7
+ # Foreign Function Interface module for Ruby.
8
+ # This is the main namespace for the Ruby-FFI library.
7
9
  module FFI
8
- # This module is just a namespace.
10
+ # BitField provides bit field functionality for Ruby-FFI.
11
+ # It allows defining, reading, and writing bit fields within FFI structs.
12
+ #
13
+ # @example Basic usage
14
+ # class MyStruct < FFI::BitStruct
15
+ # layout \
16
+ # :flags, :uint8
17
+ #
18
+ # bit_fields :flags,
19
+ # :flag1, 1,
20
+ # :flag2, 1,
21
+ # :value, 6
22
+ # end
23
+ #
24
+ # struct = MyStruct.new
25
+ # struct[:flag1] = 1
26
+ # struct[:value] = 42
9
27
  module BitField
10
28
  end
11
29
  end
@@ -6,7 +6,25 @@ require_relative 'bit_field/layout'
6
6
  require_relative 'bit_field/property'
7
7
 
8
8
  module FFI
9
- # Subclass of FFI::Struct that support bit fields.
9
+ # Subclass of FFI::Struct that supports bit fields.
10
+ # Allows defining and accessing individual bits within integer fields.
11
+ #
12
+ # @example Define a struct with bit fields
13
+ # class Flags < FFI::BitStruct
14
+ # layout \
15
+ # :value, :uint8
16
+ #
17
+ # bit_fields :value,
18
+ # :read, 1, # 1 bit for read permission
19
+ # :write, 1, # 1 bit for write permission
20
+ # :execute, 1, # 1 bit for execute permission
21
+ # :unused, 5 # 5 unused bits
22
+ # end
23
+ #
24
+ # flags = Flags.new
25
+ # flags[:read] = 1
26
+ # flags[:write] = 1
27
+ # puts flags[:value] # => 3
10
28
  class BitStruct < Struct
11
29
  # [] is defined in FFI::Struct
12
30
  alias get_member_value []
@@ -6,7 +6,28 @@ require_relative 'bit_field/layout'
6
6
  require_relative 'bit_field/property'
7
7
 
8
8
  module FFI
9
- # Subclass of FFI::ManagedStruct that support bit fields.
9
+ # Subclass of FFI::ManagedStruct that supports bit fields.
10
+ # Combines memory management with bit field functionality.
11
+ #
12
+ # Use this class when you need automatic memory management for your structs
13
+ # with bit fields. You must implement the self.release method to handle
14
+ # memory cleanup.
15
+ #
16
+ # @example Define a managed struct with bit fields
17
+ # class ManagedFlags < FFI::ManagedBitStruct
18
+ # layout \
19
+ # :value, :uint8
20
+ #
21
+ # bit_fields :value,
22
+ # :read, 1,
23
+ # :write, 1,
24
+ # :execute, 1,
25
+ # :unused, 5
26
+ #
27
+ # def self.release(ptr)
28
+ # # Custom memory cleanup code
29
+ # end
30
+ # end
10
31
  class ManagedBitStruct < ManagedStruct
11
32
  # [] is defined in FFI::Struct
12
33
  alias get_member_value []
metadata CHANGED
@@ -1,14 +1,13 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: ffi-bitfield
3
3
  version: !ruby/object:Gem::Version
4
- version: 0.0.6
4
+ version: 0.0.8
5
5
  platform: ruby
6
6
  authors:
7
7
  - kojix2
8
- autorequire:
9
8
  bindir: bin
10
9
  cert_chain: []
11
- date: 2021-07-17 00:00:00.000000000 Z
10
+ date: 2025-03-21 00:00:00.000000000 Z
12
11
  dependencies:
13
12
  - !ruby/object:Gem::Dependency
14
13
  name: ffi
@@ -24,76 +23,6 @@ dependencies:
24
23
  - - ">="
25
24
  - !ruby/object:Gem::Version
26
25
  version: '0'
27
- - !ruby/object:Gem::Dependency
28
- name: bundler
29
- requirement: !ruby/object:Gem::Requirement
30
- requirements:
31
- - - ">="
32
- - !ruby/object:Gem::Version
33
- version: '0'
34
- type: :development
35
- prerelease: false
36
- version_requirements: !ruby/object:Gem::Requirement
37
- requirements:
38
- - - ">="
39
- - !ruby/object:Gem::Version
40
- version: '0'
41
- - !ruby/object:Gem::Dependency
42
- name: irb
43
- requirement: !ruby/object:Gem::Requirement
44
- requirements:
45
- - - ">="
46
- - !ruby/object:Gem::Version
47
- version: '0'
48
- type: :development
49
- prerelease: false
50
- version_requirements: !ruby/object:Gem::Requirement
51
- requirements:
52
- - - ">="
53
- - !ruby/object:Gem::Version
54
- version: '0'
55
- - !ruby/object:Gem::Dependency
56
- name: minitest
57
- requirement: !ruby/object:Gem::Requirement
58
- requirements:
59
- - - ">="
60
- - !ruby/object:Gem::Version
61
- version: '0'
62
- type: :development
63
- prerelease: false
64
- version_requirements: !ruby/object:Gem::Requirement
65
- requirements:
66
- - - ">="
67
- - !ruby/object:Gem::Version
68
- version: '0'
69
- - !ruby/object:Gem::Dependency
70
- name: rake
71
- requirement: !ruby/object:Gem::Requirement
72
- requirements:
73
- - - ">="
74
- - !ruby/object:Gem::Version
75
- version: '0'
76
- type: :development
77
- prerelease: false
78
- version_requirements: !ruby/object:Gem::Requirement
79
- requirements:
80
- - - ">="
81
- - !ruby/object:Gem::Version
82
- version: '0'
83
- - !ruby/object:Gem::Dependency
84
- name: rubocop
85
- requirement: !ruby/object:Gem::Requirement
86
- requirements:
87
- - - ">="
88
- - !ruby/object:Gem::Version
89
- version: '0'
90
- type: :development
91
- prerelease: false
92
- version_requirements: !ruby/object:Gem::Requirement
93
- requirements:
94
- - - ">="
95
- - !ruby/object:Gem::Version
96
- version: '0'
97
26
  description: bit fields for Ruby-FFI
98
27
  email:
99
28
  - 2xijok@gmail.com
@@ -113,7 +42,6 @@ homepage: https://github.com/kojix2/ffi-bitfield
113
42
  licenses:
114
43
  - MIT
115
44
  metadata: {}
116
- post_install_message:
117
45
  rdoc_options: []
118
46
  require_paths:
119
47
  - lib
@@ -128,8 +56,7 @@ required_rubygems_version: !ruby/object:Gem::Requirement
128
56
  - !ruby/object:Gem::Version
129
57
  version: '0'
130
58
  requirements: []
131
- rubygems_version: 3.2.15
132
- signing_key:
59
+ rubygems_version: 3.6.2
133
60
  specification_version: 4
134
61
  summary: bit fields for Ruby-FFI
135
62
  test_files: []