ffi 1.15.5-x64-mingw-ucrt
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 +7 -0
- data/CHANGELOG.md +338 -0
- data/COPYING +49 -0
- data/Gemfile +14 -0
- data/LICENSE +24 -0
- data/LICENSE.SPECS +22 -0
- data/README.md +136 -0
- data/Rakefile +191 -0
- data/ffi.gemspec +42 -0
- data/lib/3.1/ffi_c.so +0 -0
- data/lib/ffi/abstract_memory.rb +44 -0
- data/lib/ffi/autopointer.rb +203 -0
- data/lib/ffi/buffer.rb +4 -0
- data/lib/ffi/callback.rb +4 -0
- data/lib/ffi/data_converter.rb +67 -0
- data/lib/ffi/enum.rb +296 -0
- data/lib/ffi/errno.rb +43 -0
- data/lib/ffi/ffi.rb +47 -0
- data/lib/ffi/io.rb +62 -0
- data/lib/ffi/library.rb +592 -0
- data/lib/ffi/managedstruct.rb +84 -0
- data/lib/ffi/memorypointer.rb +1 -0
- data/lib/ffi/platform/aarch64-darwin/types.conf +130 -0
- data/lib/ffi/platform/aarch64-freebsd/types.conf +128 -0
- data/lib/ffi/platform/aarch64-freebsd12/types.conf +181 -0
- data/lib/ffi/platform/aarch64-linux/types.conf +104 -0
- data/lib/ffi/platform/aarch64-openbsd/types.conf +134 -0
- data/lib/ffi/platform/arm-freebsd/types.conf +152 -0
- data/lib/ffi/platform/arm-freebsd12/types.conf +152 -0
- data/lib/ffi/platform/arm-linux/types.conf +132 -0
- data/lib/ffi/platform/i386-cygwin/types.conf +3 -0
- data/lib/ffi/platform/i386-darwin/types.conf +100 -0
- data/lib/ffi/platform/i386-freebsd/types.conf +152 -0
- data/lib/ffi/platform/i386-freebsd12/types.conf +152 -0
- data/lib/ffi/platform/i386-gnu/types.conf +107 -0
- data/lib/ffi/platform/i386-linux/types.conf +103 -0
- data/lib/ffi/platform/i386-netbsd/types.conf +126 -0
- data/lib/ffi/platform/i386-openbsd/types.conf +128 -0
- data/lib/ffi/platform/i386-solaris/types.conf +122 -0
- data/lib/ffi/platform/i386-windows/types.conf +52 -0
- data/lib/ffi/platform/ia64-linux/types.conf +104 -0
- data/lib/ffi/platform/mips-linux/types.conf +102 -0
- data/lib/ffi/platform/mips64-linux/types.conf +104 -0
- data/lib/ffi/platform/mips64el-linux/types.conf +104 -0
- data/lib/ffi/platform/mipsel-linux/types.conf +102 -0
- data/lib/ffi/platform/mipsisa32r6-linux/types.conf +102 -0
- data/lib/ffi/platform/mipsisa32r6el-linux/types.conf +102 -0
- data/lib/ffi/platform/mipsisa64r6-linux/types.conf +104 -0
- data/lib/ffi/platform/mipsisa64r6el-linux/types.conf +104 -0
- data/lib/ffi/platform/powerpc-aix/types.conf +180 -0
- data/lib/ffi/platform/powerpc-darwin/types.conf +100 -0
- data/lib/ffi/platform/powerpc-linux/types.conf +130 -0
- data/lib/ffi/platform/powerpc-openbsd/types.conf +156 -0
- data/lib/ffi/platform/powerpc64-linux/types.conf +104 -0
- data/lib/ffi/platform/powerpc64le-linux/types.conf +100 -0
- data/lib/ffi/platform/riscv64-linux/types.conf +104 -0
- data/lib/ffi/platform/s390-linux/types.conf +102 -0
- data/lib/ffi/platform/s390x-linux/types.conf +102 -0
- data/lib/ffi/platform/sparc-linux/types.conf +102 -0
- data/lib/ffi/platform/sparc-solaris/types.conf +128 -0
- data/lib/ffi/platform/sparc64-linux/types.conf +102 -0
- data/lib/ffi/platform/sparcv9-openbsd/types.conf +156 -0
- data/lib/ffi/platform/sparcv9-solaris/types.conf +128 -0
- data/lib/ffi/platform/x86_64-cygwin/types.conf +3 -0
- data/lib/ffi/platform/x86_64-darwin/types.conf +130 -0
- data/lib/ffi/platform/x86_64-dragonflybsd/types.conf +130 -0
- data/lib/ffi/platform/x86_64-freebsd/types.conf +128 -0
- data/lib/ffi/platform/x86_64-freebsd12/types.conf +158 -0
- data/lib/ffi/platform/x86_64-haiku/types.conf +117 -0
- data/lib/ffi/platform/x86_64-linux/types.conf +132 -0
- data/lib/ffi/platform/x86_64-msys/types.conf +119 -0
- data/lib/ffi/platform/x86_64-netbsd/types.conf +128 -0
- data/lib/ffi/platform/x86_64-openbsd/types.conf +134 -0
- data/lib/ffi/platform/x86_64-solaris/types.conf +122 -0
- data/lib/ffi/platform/x86_64-windows/types.conf +52 -0
- data/lib/ffi/platform.rb +185 -0
- data/lib/ffi/pointer.rb +167 -0
- data/lib/ffi/struct.rb +316 -0
- data/lib/ffi/struct_by_reference.rb +72 -0
- data/lib/ffi/struct_layout.rb +96 -0
- data/lib/ffi/struct_layout_builder.rb +227 -0
- data/lib/ffi/tools/const_generator.rb +232 -0
- data/lib/ffi/tools/generator.rb +105 -0
- data/lib/ffi/tools/generator_task.rb +32 -0
- data/lib/ffi/tools/struct_generator.rb +195 -0
- data/lib/ffi/tools/types_generator.rb +137 -0
- data/lib/ffi/types.rb +194 -0
- data/lib/ffi/union.rb +43 -0
- data/lib/ffi/variadic.rb +69 -0
- data/lib/ffi/version.rb +3 -0
- data/lib/ffi.rb +27 -0
- data/rakelib/ffi_gem_helper.rb +65 -0
- data/samples/getlogin.rb +8 -0
- data/samples/getpid.rb +8 -0
- data/samples/gettimeofday.rb +18 -0
- data/samples/hello.rb +8 -0
- data/samples/inotify.rb +60 -0
- data/samples/pty.rb +75 -0
- data/samples/qsort.rb +20 -0
- metadata +207 -0
data/Rakefile
ADDED
@@ -0,0 +1,191 @@
|
|
1
|
+
require 'rbconfig'
|
2
|
+
require 'date'
|
3
|
+
require 'fileutils'
|
4
|
+
require 'rbconfig'
|
5
|
+
require 'rspec/core/rake_task'
|
6
|
+
require 'rubygems/package_task'
|
7
|
+
require 'rake/extensiontask'
|
8
|
+
require_relative "lib/ffi/version"
|
9
|
+
require_relative "rakelib/ffi_gem_helper"
|
10
|
+
|
11
|
+
BUILD_DIR = "build"
|
12
|
+
BUILD_EXT_DIR = File.join(BUILD_DIR, "#{RbConfig::CONFIG['arch']}", 'ffi_c', RUBY_VERSION)
|
13
|
+
|
14
|
+
gem_spec = Bundler.load_gemspec('ffi.gemspec')
|
15
|
+
|
16
|
+
RSpec::Core::RakeTask.new(:spec => :compile) do |config|
|
17
|
+
config.rspec_opts = YAML.load_file 'spec/spec.opts'
|
18
|
+
end
|
19
|
+
|
20
|
+
desc "Build all packages"
|
21
|
+
task :package => %w[ gem:java gem:native ]
|
22
|
+
|
23
|
+
CLOBBER.include 'lib/ffi/types.conf'
|
24
|
+
CLOBBER.include 'pkg'
|
25
|
+
CLOBBER.include 'log'
|
26
|
+
|
27
|
+
CLEAN.include 'build'
|
28
|
+
CLEAN.include 'conftest.dSYM'
|
29
|
+
CLEAN.include 'spec/ffi/fixtures/libtest.{dylib,so,dll}'
|
30
|
+
CLEAN.include 'spec/ffi/fixtures/*.o'
|
31
|
+
CLEAN.include 'spec/ffi/embed-test/ext/*.{o,def}'
|
32
|
+
CLEAN.include 'spec/ffi/embed-test/ext/Makefile'
|
33
|
+
CLEAN.include "pkg/ffi-*-{mingw32,java}"
|
34
|
+
CLEAN.include 'lib/1.*'
|
35
|
+
CLEAN.include 'lib/2.*'
|
36
|
+
|
37
|
+
# clean all shipped files, that are not in git
|
38
|
+
CLEAN.include(
|
39
|
+
gem_spec.files -
|
40
|
+
`git --git-dir ext/ffi_c/libffi/.git ls-files -z`.split("\x0").map { |f| File.join("ext/ffi_c/libffi", f) } -
|
41
|
+
`git ls-files -z`.split("\x0")
|
42
|
+
)
|
43
|
+
|
44
|
+
task :distclean => :clobber
|
45
|
+
|
46
|
+
desc "Test the extension"
|
47
|
+
task :test => [ :spec ]
|
48
|
+
|
49
|
+
|
50
|
+
namespace :bench do
|
51
|
+
ITER = ENV['ITER'] ? ENV['ITER'].to_i : 100000
|
52
|
+
bench_files = Dir["bench/bench_*.rb"].sort.reject { |f| f == "bench/bench_helper.rb" }
|
53
|
+
bench_files.each do |bench|
|
54
|
+
task File.basename(bench, ".rb")[6..-1] => :compile do
|
55
|
+
sh %{#{Gem.ruby} #{bench} #{ITER}}
|
56
|
+
end
|
57
|
+
end
|
58
|
+
task :all => :compile do
|
59
|
+
bench_files.each do |bench|
|
60
|
+
sh %{#{Gem.ruby} #{bench}}
|
61
|
+
end
|
62
|
+
end
|
63
|
+
end
|
64
|
+
|
65
|
+
task 'spec:run' => :compile
|
66
|
+
task 'spec:specdoc' => :compile
|
67
|
+
|
68
|
+
task :default => :spec
|
69
|
+
|
70
|
+
namespace 'java' do
|
71
|
+
|
72
|
+
java_gem_spec = gem_spec.dup.tap do |s|
|
73
|
+
s.files.reject! { |f| File.fnmatch?("ext/*", f) }
|
74
|
+
s.extensions = []
|
75
|
+
s.platform = 'java'
|
76
|
+
end
|
77
|
+
|
78
|
+
Gem::PackageTask.new(java_gem_spec) do |pkg|
|
79
|
+
pkg.need_zip = true
|
80
|
+
pkg.need_tar = true
|
81
|
+
pkg.package_dir = 'pkg'
|
82
|
+
end
|
83
|
+
end
|
84
|
+
|
85
|
+
task 'gem:java' => 'java:gem'
|
86
|
+
|
87
|
+
FfiGemHelper.install_tasks
|
88
|
+
# Register windows gems to be pushed to rubygems.org
|
89
|
+
Bundler::GemHelper.instance.cross_platforms = %w[x86-mingw32 x64-mingw-ucrt x64-mingw32]
|
90
|
+
|
91
|
+
if RUBY_ENGINE == 'ruby' || RUBY_ENGINE == 'rbx'
|
92
|
+
require 'rake/extensiontask'
|
93
|
+
Rake::ExtensionTask.new('ffi_c', gem_spec) do |ext|
|
94
|
+
ext.name = 'ffi_c' # indicate the name of the extension.
|
95
|
+
# ext.lib_dir = BUILD_DIR # put binaries into this folder.
|
96
|
+
ext.tmp_dir = BUILD_DIR # temporary folder used during compilation.
|
97
|
+
ext.cross_compile = true # enable cross compilation (requires cross compile toolchain)
|
98
|
+
ext.cross_platform = Bundler::GemHelper.instance.cross_platforms
|
99
|
+
ext.cross_compiling do |spec|
|
100
|
+
spec.files.reject! { |path| File.fnmatch?('ext/*', path) }
|
101
|
+
end
|
102
|
+
|
103
|
+
end
|
104
|
+
else
|
105
|
+
task :compile do
|
106
|
+
STDERR.puts "Nothing to compile on #{RUBY_ENGINE}"
|
107
|
+
end
|
108
|
+
end
|
109
|
+
|
110
|
+
|
111
|
+
namespace "gem" do
|
112
|
+
task 'prepare' do
|
113
|
+
require 'rake_compiler_dock'
|
114
|
+
sh "bundle package --all"
|
115
|
+
end
|
116
|
+
|
117
|
+
Bundler::GemHelper.instance.cross_platforms.each do |plat|
|
118
|
+
desc "Build all native binary gems in parallel"
|
119
|
+
multitask 'native' => plat
|
120
|
+
|
121
|
+
desc "Build the native gem for #{plat}"
|
122
|
+
task plat => ['prepare', 'build'] do
|
123
|
+
RakeCompilerDock.sh <<-EOT, platform: plat
|
124
|
+
sudo apt-get update &&
|
125
|
+
sudo apt-get install -y libltdl-dev && bundle --local &&
|
126
|
+
rake cross native gem MAKE='nice make -j`nproc`' RUBY_CC_VERSION=${RUBY_CC_VERSION/:2.2.2/}
|
127
|
+
EOT
|
128
|
+
end
|
129
|
+
end
|
130
|
+
end
|
131
|
+
|
132
|
+
directory "ext/ffi_c/libffi"
|
133
|
+
file "ext/ffi_c/libffi/autogen.sh" => "ext/ffi_c/libffi" do
|
134
|
+
warn "Downloading libffi ..."
|
135
|
+
sh "git submodule update --init --recursive"
|
136
|
+
end
|
137
|
+
task :libffi => "ext/ffi_c/libffi/autogen.sh"
|
138
|
+
|
139
|
+
LIBFFI_GIT_FILES = `git --git-dir ext/ffi_c/libffi/.git ls-files -z`.split("\x0")
|
140
|
+
|
141
|
+
# Generate files which are in the gemspec but not in libffi's git repo by running autogen.sh
|
142
|
+
gem_spec.files.select do |f|
|
143
|
+
f =~ /ext\/ffi_c\/libffi\/(.*)/ && !LIBFFI_GIT_FILES.include?($1)
|
144
|
+
end.each do |f|
|
145
|
+
file f => "ext/ffi_c/libffi/autogen.sh" do
|
146
|
+
chdir "ext/ffi_c/libffi" do
|
147
|
+
sh "sh ./autogen.sh"
|
148
|
+
end
|
149
|
+
touch f
|
150
|
+
if gem_spec.files != Gem::Specification.load('./ffi.gemspec').files
|
151
|
+
warn "gemspec files have changed -> Please restart rake!"
|
152
|
+
exit 1
|
153
|
+
end
|
154
|
+
end
|
155
|
+
end
|
156
|
+
|
157
|
+
# Make sure we have all gemspec files before packaging
|
158
|
+
task :build => gem_spec.files
|
159
|
+
task :gem => :build
|
160
|
+
|
161
|
+
|
162
|
+
require_relative "lib/ffi/platform"
|
163
|
+
types_conf = File.expand_path(File.join(FFI::Platform::CONF_DIR, 'types.conf'))
|
164
|
+
logfile = File.join(File.dirname(__FILE__), 'types_log')
|
165
|
+
|
166
|
+
task types_conf do |task|
|
167
|
+
require 'fileutils'
|
168
|
+
require_relative "lib/ffi/tools/types_generator"
|
169
|
+
options = {}
|
170
|
+
FileUtils.mkdir_p(File.dirname(task.name), mode: 0755 )
|
171
|
+
File.open(task.name, File::CREAT|File::TRUNC|File::RDWR, 0644) do |f|
|
172
|
+
f.puts FFI::TypesGenerator.generate(options)
|
173
|
+
end
|
174
|
+
File.open(logfile, 'w') do |log|
|
175
|
+
log.puts(types_conf)
|
176
|
+
end
|
177
|
+
end
|
178
|
+
|
179
|
+
desc "Create or update type information for platform #{FFI::Platform::NAME}"
|
180
|
+
task :types_conf => types_conf
|
181
|
+
|
182
|
+
begin
|
183
|
+
require 'yard'
|
184
|
+
|
185
|
+
namespace :doc do
|
186
|
+
YARD::Rake::YardocTask.new do |yard|
|
187
|
+
end
|
188
|
+
end
|
189
|
+
rescue LoadError
|
190
|
+
warn "[warn] YARD unavailable"
|
191
|
+
end
|
data/ffi.gemspec
ADDED
@@ -0,0 +1,42 @@
|
|
1
|
+
require File.expand_path("../lib/#{File.basename(__FILE__, '.gemspec')}/version", __FILE__)
|
2
|
+
|
3
|
+
Gem::Specification.new do |s|
|
4
|
+
s.name = 'ffi'
|
5
|
+
s.version = FFI::VERSION
|
6
|
+
s.author = 'Wayne Meissner'
|
7
|
+
s.email = 'wmeissner@gmail.com'
|
8
|
+
s.homepage = 'https://github.com/ffi/ffi/wiki'
|
9
|
+
s.summary = 'Ruby FFI'
|
10
|
+
s.description = 'Ruby FFI library'
|
11
|
+
if s.respond_to?(:metadata)
|
12
|
+
s.metadata['bug_tracker_uri'] = 'https://github.com/ffi/ffi/issues'
|
13
|
+
s.metadata['changelog_uri'] = 'https://github.com/ffi/ffi/blob/master/CHANGELOG.md'
|
14
|
+
s.metadata['documentation_uri'] = 'https://github.com/ffi/ffi/wiki'
|
15
|
+
s.metadata['wiki_uri'] = 'https://github.com/ffi/ffi/wiki'
|
16
|
+
s.metadata['source_code_uri'] = 'https://github.com/ffi/ffi/'
|
17
|
+
s.metadata['mailing_list_uri'] = 'http://groups.google.com/group/ruby-ffi'
|
18
|
+
end
|
19
|
+
s.files = `git ls-files -z`.split("\x0").reject do |f|
|
20
|
+
f =~ /^(\.|bench|gen|libtest|nbproject|spec)/
|
21
|
+
end
|
22
|
+
|
23
|
+
# Add libffi git files
|
24
|
+
lfs = `git --git-dir ext/ffi_c/libffi/.git ls-files -z`.split("\x0")
|
25
|
+
# Add autoconf generated files of libffi
|
26
|
+
lfs += %w[ configure config.guess config.sub install-sh ltmain.sh missing fficonfig.h.in ]
|
27
|
+
# Add automake generated files of libffi
|
28
|
+
lfs += `git --git-dir ext/ffi_c/libffi/.git ls-files -z *.am */*.am`.gsub(".am\0", ".in\0").split("\x0")
|
29
|
+
s.files += lfs.map do |f|
|
30
|
+
File.join("ext/ffi_c/libffi", f)
|
31
|
+
end
|
32
|
+
|
33
|
+
s.extensions << 'ext/ffi_c/extconf.rb'
|
34
|
+
s.rdoc_options = %w[--exclude=ext/ffi_c/.*\.o$ --exclude=ffi_c\.(bundle|so)$]
|
35
|
+
s.license = 'BSD-3-Clause'
|
36
|
+
s.require_paths << 'ext/ffi_c'
|
37
|
+
s.required_ruby_version = '>= 2.3'
|
38
|
+
s.add_development_dependency 'rake', '~> 13.0'
|
39
|
+
s.add_development_dependency 'rake-compiler', '~> 1.0'
|
40
|
+
s.add_development_dependency 'rake-compiler-dock', '~> 1.0'
|
41
|
+
s.add_development_dependency 'rspec', '~> 2.14.1'
|
42
|
+
end
|
data/lib/3.1/ffi_c.so
ADDED
Binary file
|
@@ -0,0 +1,44 @@
|
|
1
|
+
#
|
2
|
+
# Copyright (C) 2020 Lars Kanis
|
3
|
+
#
|
4
|
+
# This file is part of ruby-ffi.
|
5
|
+
#
|
6
|
+
# All rights reserved.
|
7
|
+
#
|
8
|
+
# Redistribution and use in source and binary forms, with or without
|
9
|
+
# modification, are permitted provided that the following conditions are met:
|
10
|
+
#
|
11
|
+
# * Redistributions of source code must retain the above copyright notice, this
|
12
|
+
# list of conditions and the following disclaimer.
|
13
|
+
# * Redistributions in binary form must reproduce the above copyright notice
|
14
|
+
# this list of conditions and the following disclaimer in the documentation
|
15
|
+
# and/or other materials provided with the distribution.
|
16
|
+
# * Neither the name of the Ruby FFI project nor the names of its contributors
|
17
|
+
# may be used to endorse or promote products derived from this software
|
18
|
+
# without specific prior written permission.
|
19
|
+
#
|
20
|
+
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
21
|
+
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
22
|
+
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
23
|
+
# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
|
24
|
+
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
25
|
+
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
26
|
+
# SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
27
|
+
# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
28
|
+
# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
29
|
+
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.#
|
30
|
+
|
31
|
+
|
32
|
+
module FFI
|
33
|
+
class AbstractMemory
|
34
|
+
LONG_MAX = FFI::Pointer.new(1).size
|
35
|
+
private_constant :LONG_MAX
|
36
|
+
|
37
|
+
# Return +true+ if +self+ has a size limit.
|
38
|
+
#
|
39
|
+
# @return [Boolean]
|
40
|
+
def size_limit?
|
41
|
+
size != LONG_MAX
|
42
|
+
end
|
43
|
+
end
|
44
|
+
end
|
@@ -0,0 +1,203 @@
|
|
1
|
+
#
|
2
|
+
# Copyright (C) 2008-2010 Wayne Meissner
|
3
|
+
# Copyright (C) 2008 Mike Dalessio
|
4
|
+
#
|
5
|
+
# This file is part of ruby-ffi.
|
6
|
+
#
|
7
|
+
# All rights reserved.
|
8
|
+
#
|
9
|
+
# Redistribution and use in source and binary forms, with or without
|
10
|
+
# modification, are permitted provided that the following conditions are met:
|
11
|
+
#
|
12
|
+
# * Redistributions of source code must retain the above copyright notice, this
|
13
|
+
# list of conditions and the following disclaimer.
|
14
|
+
# * Redistributions in binary form must reproduce the above copyright notice
|
15
|
+
# this list of conditions and the following disclaimer in the documentation
|
16
|
+
# and/or other materials provided with the distribution.
|
17
|
+
# * Neither the name of the Ruby FFI project nor the names of its contributors
|
18
|
+
# may be used to endorse or promote products derived from this software
|
19
|
+
# without specific prior written permission.
|
20
|
+
#
|
21
|
+
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
22
|
+
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
23
|
+
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
24
|
+
# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
|
25
|
+
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
26
|
+
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
27
|
+
# SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
28
|
+
# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
29
|
+
# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
30
|
+
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
31
|
+
|
32
|
+
module FFI
|
33
|
+
class AutoPointer < Pointer
|
34
|
+
extend DataConverter
|
35
|
+
|
36
|
+
# @overload initialize(pointer, method)
|
37
|
+
# @param pointer [Pointer]
|
38
|
+
# @param method [Method]
|
39
|
+
# @return [self]
|
40
|
+
# The passed Method will be invoked at GC time.
|
41
|
+
# @overload initialize(pointer, proc)
|
42
|
+
# @param pointer [Pointer]
|
43
|
+
# @return [self]
|
44
|
+
# The passed Proc will be invoked at GC time (SEE WARNING BELOW!)
|
45
|
+
# @note WARNING: passing a proc _may_ cause your pointer to never be
|
46
|
+
# GC'd, unless you're careful to avoid trapping a reference to the
|
47
|
+
# pointer in the proc. See the test specs for examples.
|
48
|
+
# @overload initialize(pointer) { |p| ... }
|
49
|
+
# @param pointer [Pointer]
|
50
|
+
# @yieldparam [Pointer] p +pointer+ passed to the block
|
51
|
+
# @return [self]
|
52
|
+
# The passed block will be invoked at GC time.
|
53
|
+
# @note
|
54
|
+
# WARNING: passing a block will cause your pointer to never be GC'd.
|
55
|
+
# This is bad.
|
56
|
+
# @overload initialize(pointer)
|
57
|
+
# @param pointer [Pointer]
|
58
|
+
# @return [self]
|
59
|
+
# The pointer's release() class method will be invoked at GC time.
|
60
|
+
#
|
61
|
+
# @note The safest, and therefore preferred, calling
|
62
|
+
# idiom is to pass a Method as the second parameter. Example usage:
|
63
|
+
#
|
64
|
+
# class PointerHelper
|
65
|
+
# def self.release(pointer)
|
66
|
+
# ...
|
67
|
+
# end
|
68
|
+
# end
|
69
|
+
#
|
70
|
+
# p = AutoPointer.new(other_pointer, PointerHelper.method(:release))
|
71
|
+
#
|
72
|
+
# The above code will cause PointerHelper#release to be invoked at GC time.
|
73
|
+
#
|
74
|
+
# @note
|
75
|
+
# The last calling idiom (only one parameter) is generally only
|
76
|
+
# going to be useful if you subclass {AutoPointer}, and override
|
77
|
+
# #release, which by default does nothing.
|
78
|
+
def initialize(ptr, proc=nil, &block)
|
79
|
+
super(ptr.type_size, ptr)
|
80
|
+
raise TypeError, "Invalid pointer" if ptr.nil? || !ptr.kind_of?(Pointer) \
|
81
|
+
|| ptr.kind_of?(MemoryPointer) || ptr.kind_of?(AutoPointer)
|
82
|
+
|
83
|
+
@releaser = if proc
|
84
|
+
if not proc.respond_to?(:call)
|
85
|
+
raise RuntimeError.new("proc must be callable")
|
86
|
+
end
|
87
|
+
CallableReleaser.new(ptr, proc)
|
88
|
+
|
89
|
+
else
|
90
|
+
if not self.class.respond_to?(:release)
|
91
|
+
raise RuntimeError.new("no release method defined")
|
92
|
+
end
|
93
|
+
DefaultReleaser.new(ptr, self.class)
|
94
|
+
end
|
95
|
+
|
96
|
+
ObjectSpace.define_finalizer(self, @releaser)
|
97
|
+
self
|
98
|
+
end
|
99
|
+
|
100
|
+
# @return [nil]
|
101
|
+
# Free the pointer.
|
102
|
+
def free
|
103
|
+
@releaser.free
|
104
|
+
end
|
105
|
+
|
106
|
+
# @param [Boolean] autorelease
|
107
|
+
# @return [Boolean] +autorelease+
|
108
|
+
# Set +autorelease+ property. See {Pointer Autorelease section at Pointer}.
|
109
|
+
def autorelease=(autorelease)
|
110
|
+
@releaser.autorelease=(autorelease)
|
111
|
+
end
|
112
|
+
|
113
|
+
# @return [Boolean] +autorelease+
|
114
|
+
# Get +autorelease+ property. See {Pointer Autorelease section at Pointer}.
|
115
|
+
def autorelease?
|
116
|
+
@releaser.autorelease
|
117
|
+
end
|
118
|
+
|
119
|
+
# @abstract Base class for {AutoPointer}'s releasers.
|
120
|
+
#
|
121
|
+
# All subclasses of Releaser should define a +#release(ptr)+ method.
|
122
|
+
# A releaser is an object in charge of release an {AutoPointer}.
|
123
|
+
class Releaser
|
124
|
+
attr_accessor :autorelease
|
125
|
+
|
126
|
+
# @param [Pointer] ptr
|
127
|
+
# @param [#call] proc
|
128
|
+
# @return [nil]
|
129
|
+
# A new instance of Releaser.
|
130
|
+
def initialize(ptr, proc)
|
131
|
+
@ptr = ptr
|
132
|
+
@proc = proc
|
133
|
+
@autorelease = true
|
134
|
+
end
|
135
|
+
|
136
|
+
# @return [nil]
|
137
|
+
# Free pointer.
|
138
|
+
def free
|
139
|
+
if @ptr
|
140
|
+
release(@ptr)
|
141
|
+
@autorelease = false
|
142
|
+
@ptr = nil
|
143
|
+
@proc = nil
|
144
|
+
end
|
145
|
+
end
|
146
|
+
|
147
|
+
# @param args
|
148
|
+
# Release pointer if +autorelease+ is set.
|
149
|
+
def call(*args)
|
150
|
+
release(@ptr) if @autorelease && @ptr
|
151
|
+
end
|
152
|
+
end
|
153
|
+
|
154
|
+
# DefaultReleaser is a {Releaser} used when an {AutoPointer} is defined
|
155
|
+
# without Proc or Method. In this case, the pointer to release must be of
|
156
|
+
# a class derived from AutoPointer with a {release} class method.
|
157
|
+
class DefaultReleaser < Releaser
|
158
|
+
# @param [Pointer] ptr
|
159
|
+
# @return [nil]
|
160
|
+
# Release +ptr+ using the {release} class method of its class.
|
161
|
+
def release(ptr)
|
162
|
+
@proc.release(ptr)
|
163
|
+
end
|
164
|
+
end
|
165
|
+
|
166
|
+
# CallableReleaser is a {Releaser} used when an {AutoPointer} is defined with a
|
167
|
+
# Proc or a Method.
|
168
|
+
class CallableReleaser < Releaser
|
169
|
+
# Release +ptr+ by using Proc or Method defined at +ptr+
|
170
|
+
# {AutoPointer#initialize initialization}.
|
171
|
+
#
|
172
|
+
# @param [Pointer] ptr
|
173
|
+
# @return [nil]
|
174
|
+
def release(ptr)
|
175
|
+
@proc.call(ptr)
|
176
|
+
end
|
177
|
+
end
|
178
|
+
|
179
|
+
# Return native type of AutoPointer.
|
180
|
+
#
|
181
|
+
# Override {DataConverter#native_type}.
|
182
|
+
# @return [Type::POINTER]
|
183
|
+
# @raise {RuntimeError} if class does not implement a +#release+ method
|
184
|
+
def self.native_type
|
185
|
+
if not self.respond_to?(:release)
|
186
|
+
raise RuntimeError.new("no release method defined for #{self.inspect}")
|
187
|
+
end
|
188
|
+
Type::POINTER
|
189
|
+
end
|
190
|
+
|
191
|
+
# Create a new AutoPointer.
|
192
|
+
#
|
193
|
+
# Override {DataConverter#from_native}.
|
194
|
+
# @overload self.from_native(ptr, ctx)
|
195
|
+
# @param [Pointer] ptr
|
196
|
+
# @param ctx not used. Please set +nil+.
|
197
|
+
# @return [AutoPointer]
|
198
|
+
def self.from_native(val, ctx)
|
199
|
+
self.new(val)
|
200
|
+
end
|
201
|
+
end
|
202
|
+
|
203
|
+
end
|
data/lib/ffi/buffer.rb
ADDED
data/lib/ffi/callback.rb
ADDED
@@ -0,0 +1,67 @@
|
|
1
|
+
#
|
2
|
+
# Copyright (C) 2008-2010 Wayne Meissner
|
3
|
+
#
|
4
|
+
# This file is part of ruby-ffi.
|
5
|
+
#
|
6
|
+
# All rights reserved.
|
7
|
+
#
|
8
|
+
# Redistribution and use in source and binary forms, with or without
|
9
|
+
# modification, are permitted provided that the following conditions are met:
|
10
|
+
#
|
11
|
+
# * Redistributions of source code must retain the above copyright notice, this
|
12
|
+
# list of conditions and the following disclaimer.
|
13
|
+
# * Redistributions in binary form must reproduce the above copyright notice
|
14
|
+
# this list of conditions and the following disclaimer in the documentation
|
15
|
+
# and/or other materials provided with the distribution.
|
16
|
+
# * Neither the name of the Ruby FFI project nor the names of its contributors
|
17
|
+
# may be used to endorse or promote products derived from this software
|
18
|
+
# without specific prior written permission.
|
19
|
+
#
|
20
|
+
# THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
|
21
|
+
# AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
|
22
|
+
# IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
|
23
|
+
# DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
|
24
|
+
# FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
25
|
+
# DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
26
|
+
# SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
27
|
+
# CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
28
|
+
# OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
29
|
+
# OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.#
|
30
|
+
|
31
|
+
module FFI
|
32
|
+
# This module is used to extend somes classes and give then a common API.
|
33
|
+
#
|
34
|
+
# Most of methods defined here must be overriden.
|
35
|
+
module DataConverter
|
36
|
+
# Get native type.
|
37
|
+
#
|
38
|
+
# @overload native_type(type)
|
39
|
+
# @param [String, Symbol, Type] type
|
40
|
+
# @return [Type]
|
41
|
+
# Get native type from +type+.
|
42
|
+
#
|
43
|
+
# @overload native_type
|
44
|
+
# @raise {NotImplementedError} This method must be overriden.
|
45
|
+
def native_type(type = nil)
|
46
|
+
if type
|
47
|
+
@native_type = FFI.find_type(type)
|
48
|
+
else
|
49
|
+
native_type = @native_type
|
50
|
+
unless native_type
|
51
|
+
raise NotImplementedError, 'native_type method not overridden and no native_type set'
|
52
|
+
end
|
53
|
+
native_type
|
54
|
+
end
|
55
|
+
end
|
56
|
+
|
57
|
+
# Convert to a native type.
|
58
|
+
def to_native(value, ctx)
|
59
|
+
value
|
60
|
+
end
|
61
|
+
|
62
|
+
# Convert from a native type.
|
63
|
+
def from_native(value, ctx)
|
64
|
+
value
|
65
|
+
end
|
66
|
+
end
|
67
|
+
end
|