class FFI::Generator
Generate files with C structs for FFI::Struct
and C constants.
A simple example¶ ↑
In file zlib.rb.ffi
:
module Zlib @@@ constants do |c| c.include "zlib.h" c.const :ZLIB_VERNUM end @@@ class ZStream < FFI::Struct struct do |s| s.name "struct z_stream_s" s.include "zlib.h" s.field :next_in, :pointer s.field :avail_in, :uint s.field :total_in, :ulong end @@@ end end
Translate the file:
require "ffi/tools/generator" FFI::Generator.new "zlib.rb.ffi", "zlib.rb"
Generates the file zlib.rb
with constant values and offsets:
module Zlib ZLIB_VERNUM = 4784 class ZStream < FFI::Struct layout :next_in, :pointer, 0, :avail_in, :uint, 8, :total_in, :ulong, 16 end
@see FFI::Generator::Task
for easy integration in a Rakefile
Public Class Methods
# File lib/ffi/tools/generator.rb, line 51 def initialize(ffi_name, rb_name, options = {}) @ffi_name = ffi_name @rb_name = rb_name @options = options @name = File.basename rb_name, '.rb' file = File.read @ffi_name new_file = file.gsub(/^( *)@@@(.*?)@@@/m) do @constants = [] @structs = [] indent = $1 original_lines = $2.count "\n" instance_eval $2, @ffi_name, $`.count("\n") new_lines = [] @constants.each { |c| new_lines << c.to_ruby } @structs.each { |s| new_lines << s.generate_layout } new_lines = new_lines.join("\n").split "\n" # expand multiline blocks new_lines = new_lines.map { |line| indent + line } padding = original_lines - new_lines.length new_lines += [nil] * padding if padding >= 0 new_lines.join "\n" end open @rb_name, 'w' do |f| f.puts "# This file is generated from `#{@ffi_name}'. Do not edit." f.puts f.puts new_file end end
Public Instance Methods
# File lib/ffi/tools/generator.rb, line 88 def constants(options = {}, &block) @constants << FFI::ConstGenerator.new(@name, @options.merge(options), &block) end
# File lib/ffi/tools/generator.rb, line 92 def struct(options = {}, &block) @structs << FFI::StructGenerator.new(@name, @options.merge(options), &block) end
Utility converter for constants
# File lib/ffi/tools/generator.rb, line 99 def to_s proc { |obj| obj.to_s.inspect } end