2017-05-07 08:04:49 -04:00
|
|
|
require File.expand_path('../../../spec_helper', __FILE__)
|
2017-06-15 08:48:52 -04:00
|
|
|
|
|
|
|
# MRI magic to use built but not installed ruby
|
2017-05-28 12:40:12 -04:00
|
|
|
$extmk = false
|
2017-05-07 08:04:49 -04:00
|
|
|
|
|
|
|
require 'rbconfig'
|
|
|
|
|
2017-06-15 08:48:52 -04:00
|
|
|
OBJDIR ||= File.expand_path("../../../ext/#{RUBY_ENGINE}/#{RUBY_VERSION}", __FILE__)
|
2017-05-14 10:09:56 -04:00
|
|
|
mkdir_p(OBJDIR)
|
2017-05-07 08:04:49 -04:00
|
|
|
|
|
|
|
def extension_path
|
|
|
|
File.expand_path("../ext", __FILE__)
|
|
|
|
end
|
|
|
|
|
|
|
|
def object_path
|
|
|
|
OBJDIR
|
|
|
|
end
|
|
|
|
|
|
|
|
def compile_extension(name)
|
2017-06-15 08:48:52 -04:00
|
|
|
debug = false
|
|
|
|
run_mkmf_in_process = RUBY_ENGINE == 'truffleruby'
|
2017-05-07 08:04:49 -04:00
|
|
|
|
2017-06-15 08:48:52 -04:00
|
|
|
ext = "#{name}_spec"
|
|
|
|
lib = "#{object_path}/#{ext}.#{RbConfig::CONFIG['DLEXT']}"
|
|
|
|
ruby_header = "#{RbConfig::CONFIG['rubyhdrdir']}/ruby.h"
|
2017-10-28 11:15:48 -04:00
|
|
|
libruby_so = RbConfig::CONFIG['LIBRUBY_SO']
|
|
|
|
ruby_library = "#{RbConfig::CONFIG['libdir']}/#{libruby_so}"
|
|
|
|
unless libruby_so and File.exist?(ruby_library)
|
2017-10-28 16:04:52 -04:00
|
|
|
# Statically-compiled lib in the binary, ignore this check
|
|
|
|
ruby_library = nil
|
2017-10-28 11:15:48 -04:00
|
|
|
end
|
2017-05-28 12:40:12 -04:00
|
|
|
|
2017-06-15 08:48:52 -04:00
|
|
|
return lib if File.exist?(lib) and
|
|
|
|
File.mtime(lib) > File.mtime("#{extension_path}/rubyspec.h") and
|
|
|
|
File.mtime(lib) > File.mtime("#{extension_path}/#{ext}.c") and
|
2017-05-28 08:02:08 -04:00
|
|
|
File.mtime(lib) > File.mtime(ruby_header) and
|
2017-10-28 16:04:52 -04:00
|
|
|
(!ruby_library || File.mtime(lib) > File.mtime(ruby_library)) and
|
2017-05-07 08:04:49 -04:00
|
|
|
true # sentinel
|
|
|
|
|
2017-05-27 17:55:02 -04:00
|
|
|
# Copy needed source files to tmpdir
|
|
|
|
tmpdir = tmp("cext_#{name}")
|
2017-06-15 08:48:52 -04:00
|
|
|
Dir.mkdir(tmpdir)
|
2017-05-07 08:04:49 -04:00
|
|
|
begin
|
2017-06-15 08:48:52 -04:00
|
|
|
["rubyspec.h", "#{ext}.c"].each do |file|
|
|
|
|
cp "#{extension_path}/#{file}", "#{tmpdir}/#{file}"
|
2017-05-27 17:55:02 -04:00
|
|
|
end
|
|
|
|
|
|
|
|
Dir.chdir(tmpdir) do
|
2017-06-15 08:48:52 -04:00
|
|
|
if run_mkmf_in_process
|
|
|
|
required = require 'mkmf'
|
|
|
|
# Reinitialize mkmf if already required
|
|
|
|
init_mkmf unless required
|
|
|
|
create_makefile(ext, tmpdir)
|
|
|
|
else
|
|
|
|
File.write("extconf.rb", "require 'mkmf'\n" +
|
|
|
|
"$ruby = ENV.values_at('RUBY_EXE', 'RUBY_FLAGS').join(' ')\n" +
|
|
|
|
# MRI magic to consider building non-bundled extensions
|
|
|
|
"$extout = nil\n" +
|
|
|
|
"create_makefile(#{ext.inspect})\n")
|
|
|
|
output = ruby_exe("extconf.rb")
|
|
|
|
raise "extconf failed:\n#{output}" unless $?.success?
|
|
|
|
$stderr.puts output if debug
|
2017-05-28 12:29:04 -04:00
|
|
|
end
|
2017-06-15 08:48:52 -04:00
|
|
|
|
|
|
|
# Do not capture stderr as we want to show compiler warnings
|
2017-09-20 15:45:19 -04:00
|
|
|
make, opts = setup_make
|
2017-08-02 20:53:25 -04:00
|
|
|
output = IO.popen([make, "V=1", "DESTDIR=", opts], &:read)
|
2017-06-15 08:48:52 -04:00
|
|
|
raise "#{make} failed:\n#{output}" unless $?.success?
|
|
|
|
$stderr.puts output if debug
|
|
|
|
|
|
|
|
cp File.basename(lib), lib
|
2017-05-07 08:04:49 -04:00
|
|
|
end
|
|
|
|
ensure
|
2017-05-27 17:55:02 -04:00
|
|
|
rm_r tmpdir
|
2017-05-07 08:04:49 -04:00
|
|
|
end
|
2017-05-27 17:55:02 -04:00
|
|
|
|
2017-06-15 08:48:52 -04:00
|
|
|
File.chmod(0755, lib)
|
2017-05-27 17:55:02 -04:00
|
|
|
lib
|
2017-05-07 08:04:49 -04:00
|
|
|
end
|
|
|
|
|
2017-09-20 15:45:19 -04:00
|
|
|
def setup_make
|
|
|
|
make = ENV['MAKE']
|
|
|
|
make ||= (RbConfig::CONFIG['host_os'].include?("mswin") ? "nmake" : "make")
|
|
|
|
make_flags = ENV["MAKEFLAGS"] || ''
|
|
|
|
|
|
|
|
# suppress logo of nmake.exe to stderr
|
|
|
|
if File.basename(make, ".*").downcase == "nmake" and !make_flags.include?("l")
|
|
|
|
ENV["MAKEFLAGS"] = "l#{make_flags}"
|
|
|
|
end
|
|
|
|
|
|
|
|
opts = {}
|
|
|
|
if /(?:\A|\s)--jobserver-(?:auth|fds)=(\d+),(\d+)/ =~ make_flags
|
|
|
|
begin
|
|
|
|
r = IO.for_fd($1.to_i(10), "rb", autoclose: false)
|
|
|
|
w = IO.for_fd($2.to_i(10), "wb", autoclose: false)
|
|
|
|
rescue Errno::EBADF
|
|
|
|
else
|
|
|
|
opts[r] = r
|
|
|
|
opts[w] = w
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
[make, opts]
|
|
|
|
end
|
|
|
|
|
2017-05-07 08:04:49 -04:00
|
|
|
def load_extension(name)
|
|
|
|
require compile_extension(name)
|
|
|
|
rescue LoadError
|
|
|
|
if %r{/usr/sbin/execerror ruby "\(ld 3 1 main ([/a-zA-Z0-9_\-.]+_spec\.so)"} =~ $!.message
|
|
|
|
system('/usr/sbin/execerror', "#{RbConfig::CONFIG["bindir"]}/ruby", "(ld 3 1 main #{$1}")
|
|
|
|
end
|
|
|
|
raise
|
|
|
|
end
|
|
|
|
|
|
|
|
# Constants
|
|
|
|
CAPI_SIZEOF_LONG = [0].pack('l!').size
|